/* ════════════════════════════════════════════════════════════════════
   Breakpoints canoniques (utiliser uniquement ces 2) — v1.10.0
   max-width: 480px   -> phone (touch targets 44px, font ≥ 16px)
   max-width: 720px   -> tablette / topbar mobile (hamburger, nav stack)
   Tout autre breakpoint est une erreur de cohérence à corriger.
   ════════════════════════════════════════════════════════════════════ */

/* ─── Theme tokens (CSS custom properties) ──────────────────────── */
/* v1.10.0 : les 6 presets de couleur sont déclarés en CSS via
   `:root[data-color="..."]` + variante `[data-scheme="dark"]`. Les valeurs
   `--color-primary*` du `:root` nu ne servent que de fallback si data-color
   est absent (cas: app très ancienne pré-v1.10.0). theme-boot.js ne calcule
   plus que pour le cas `data-color="custom"` (hex utilisateur). Apps :
   utiliser ces tokens via var() au lieu de couleurs en dur. */
:root {
  /* ─ Fallback (utilisé si <html data-color> absent — pré-v1.10.0) ─ */
  --color-primary: #1d4ed8;
  --color-primary-dark: #1e40af;
  --color-primary-bg: #eff6ff;
  --color-primary-border: #93c5fd;

  /* ─ Typo ─ */
  --font-scale: 1;
  --font-family: system-ui, -apple-system, "Segoe UI", Roboto, sans-serif;
  --font-size-xs: 0.75rem;
  --font-size-sm: 0.85rem;
  --font-size-md: 0.95rem;
  --font-size-lg: 1.15rem;
  --font-size-xl: 1.5rem;

  /* ─ Couleurs neutres ─ */
  --bg: #fafaf7;
  --fg: #1f2937;
  --fg-muted: #6b7280;
  --card-bg: #ffffff;
  --surface-2: #f3f4f6;
  --surface-3: #f9fafb;
  --border: #e5e7eb;
  --border-strong: #d1d5db;

  /* ─ Sémantiques (success / warn / error / info) ─ */
  --color-success: #16a34a;
  --color-success-bg: #f0fdf4;
  --color-success-border: #bbf7d0;
  --color-warn: #ea580c;
  --color-warn-bg: #fff7ed;
  --color-warn-border: #fed7aa;
  --color-error: #dc2626;
  --color-error-bg: #fef2f2;
  --color-error-border: #fecaca;
  --color-info: #0284c7;
  --color-info-bg: #f0f9ff;
  --color-info-border: #bae6fd;

  /* ─ Spacing & rythme ─ */
  --pad-y: 0.55rem;
  --pad-x: 0.75rem;
  --gap: 0.5rem;
  --gap-sm: 0.35rem;
  --gap-md: 0.75rem;
  --gap-lg: 1.25rem;

  /* ─ Rayons ─ */
  --radius-sm: 4px;
  --radius-md: 6px;
  --radius-lg: 10px;
  --radius-pill: 999px;

  /* ─ Ombres (douces, app interne, pas tape-à-l'œil) ─ */
  --shadow-sm: 0 1px 2px rgba(0,0,0,0.04);
  --shadow-md: 0 2px 8px rgba(0,0,0,0.06);
  --shadow-lg: 0 8px 24px rgba(0,0,0,0.10);

  /* ─ Focus ring (accessibilité — visible mais pas agressif) ─ */
  --focus-ring: 0 0 0 3px var(--color-primary-bg);
  --focus-ring-strong: 0 0 0 3px rgba(29, 78, 216, 0.25);
}
:root[data-scheme="dark"] {
  --bg: #0f172a;
  --fg: #e5e7eb;
  --fg-muted: #94a3b8;
  --card-bg: #1e293b;
  --surface-2: #1e293b;
  --surface-3: #172033;
  --border: #334155;
  --border-strong: #475569;

  /* Sémantiques en dark : versions plus sombres pour les fonds */
  --color-success-bg: rgba(22, 163, 74, 0.12);
  --color-warn-bg: rgba(234, 88, 12, 0.12);
  --color-error-bg: rgba(220, 38, 38, 0.12);
  --color-info-bg: rgba(2, 132, 199, 0.12);

  --focus-ring: 0 0 0 3px rgba(29, 78, 216, 0.35);
}
:root[data-density="compact"] {
  --pad-y: 0.35rem;
  --pad-x: 0.55rem;
  --gap: 0.35rem;
}

/* ─── Color presets (v1.10.0) ──────────────────────────────────────
   Une paire light/dark par preset. La variante dark surcharge uniquement
   `--color-primary-bg` (les autres tons restent identiques) : le bg pâle
   (~95% lightness) devient illisible avec le texte clair du dark mode,
   donc on le remplace par un fond semi-transparent saturé. Sélecteurs
   d'attribut → 100% CSS, fonctionne même si theme-boot.js est bloqué. */
:root[data-color="blue"] {
  --color-primary: #1d4ed8;
  --color-primary-dark: #1e40af;
  --color-primary-bg: #eff6ff;
  --color-primary-border: #93c5fd;
}
:root[data-color="blue"][data-scheme="dark"] {
  --color-primary-bg: rgba(59, 130, 246, 0.18);
}

:root[data-color="green"] {
  --color-primary: #059669;
  --color-primary-dark: #047857;
  --color-primary-bg: #ecfdf5;
  --color-primary-border: #6ee7b7;
}
:root[data-color="green"][data-scheme="dark"] {
  --color-primary-bg: rgba(16, 185, 129, 0.18);
}

:root[data-color="violet"] {
  --color-primary: #7c3aed;
  --color-primary-dark: #6d28d9;
  --color-primary-bg: #f5f3ff;
  --color-primary-border: #c4b5fd;
}
:root[data-color="violet"][data-scheme="dark"] {
  --color-primary-bg: rgba(139, 92, 246, 0.20);
}

:root[data-color="orange"] {
  --color-primary: #ea580c;
  --color-primary-dark: #c2410c;
  --color-primary-bg: #fff7ed;
  --color-primary-border: #fdba74;
}
:root[data-color="orange"][data-scheme="dark"] {
  --color-primary-bg: rgba(249, 115, 22, 0.20);
}

:root[data-color="teal"] {
  --color-primary: #0d9488;
  --color-primary-dark: #0f766e;
  --color-primary-bg: #f0fdfa;
  --color-primary-border: #5eead4;
}
:root[data-color="teal"][data-scheme="dark"] {
  --color-primary-bg: rgba(20, 184, 166, 0.18);
}

:root[data-color="slate"] {
  --color-primary: #475569;
  --color-primary-dark: #334155;
  --color-primary-bg: #f1f5f9;
  --color-primary-border: #cbd5e1;
}
:root[data-color="slate"][data-scheme="dark"] {
  --color-primary-bg: rgba(148, 163, 184, 0.18);
}

/* ─── Reset + base ──────────────────────────────────────────────── */
* { box-sizing: border-box; }
html, body { margin: 0; padding: 0; }
/* Garde-fou HTML5 : l'attribut [hidden] doit toujours masquer l'element.
   Le UA stylesheet pose `[hidden] { display: none }` mais avec specificite
   egale (0,1,0) une regle app comme `.foo { display: flex }` la perd dans
   la cascade (l'ordre source gagne). Sans ce !important, toggler
   element.hidden = true peut ne pas masquer l'element. Spec-conforme. */
[hidden] { display: none !important; }
body {
  font-family: var(--font-family);
  color: var(--fg);
  background: var(--bg);
  line-height: 1.4;
  font-size: calc(1rem * var(--font-scale));
}
/* Form controls don't inherit font by default in some browsers — force it
   so the user's font choice actually applies to inputs, selects, buttons.
   Le `max(16px, …)` empêche le zoom auto iOS Safari sur focus quand le
   thème ou la densité compacte a réduit la font-size sous 16px : iOS
   considère <16px comme "trop petit" et zoome automatiquement, ce qui
   ruine la mise en page mobile. */
input, select, textarea, button {
  font-family: inherit;
  font-size: max(16px, 1em);
}
a { color: var(--color-primary); text-decoration: none; }
a:hover { text-decoration: underline; }

main { max-width: 980px; margin: 1.5rem auto; padding: 0 1rem; }
/* v1.4.4 : sur les pages /settings/*, on libère le max-width.
   L'admin est dense (forms, cartes, controles), pas du texte courant —
   la contrainte de lisibilite n'a pas d'interet ici, alors qu'elle
   genere beaucoup d'espace vide sur grand ecran. La sidebar
   .settings-nav garde sa largeur fixe, seul le contenu s'etale. */
main:has(.settings-layout) { max-width: 1400px; }
section { margin-bottom: 2rem; }
h1, h2 { color: #111827; }
h1 { font-size: 1.5rem; margin-top: 0; }
h2 { font-size: 1.15rem; margin-bottom: 0.75rem; }

/* ─── Topbar (sticky) ───────────────────────────────────────────── */
/* Background : depuis v1.3.0, suit le skin (var --color-primary-dark)
   par défaut, en clair ET en sombre. L'utilisateur peut forcer une
   couleur différente via /settings/apparence (option `topbar` = `skin`
   par défaut, `neutral` pour un gris foncé indépendant du skin, ou
   `custom` pour une pipette). theme-boot.js applique le résultat dans
   --topbar-bg ; sinon fallback sur --color-primary-dark. */
.topbar {
  display: flex;
  align-items: center;
  gap: 1rem;
  padding: 0.6rem 1rem;
  background: var(--topbar-bg, var(--color-primary-dark));
  color: #fff;
  position: sticky;
  top: 0;
  z-index: 50;
  box-shadow: 0 2px 8px rgba(0,0,0,0.15);
  transition: background-color 0.15s ease;
}
/* v1.9.0 : option utilisateur — topbar défile avec la page au lieu
   d'être ancrée en haut. Posé via le cookie `<prefix>_theme_topbar_position`
   et l'attribut `data-topbar-position="static"` sur <html> par theme-boot.js. */
:root[data-topbar-position="static"] .topbar {
  position: static;
}
.topbar .brand {
  color: #fff;
  font-weight: 700;
  font-size: 1.1rem;
  text-decoration: none;
  white-space: nowrap;
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
}
.brand-icon {
  width: 26px;
  height: 26px;
  border-radius: 5px;
  vertical-align: middle;
  background: rgba(255,255,255,0.05);
}
.brand-version {
  font-weight: 400;
  font-size: 0.75rem;
  margin-left: 0.35rem;
  padding: 0.1rem 0.4rem;
  background: rgba(255,255,255,0.15);
  border-radius: 999px;
  color: #d1d5db;
  vertical-align: middle;
}
.topbar .nav {
  display: flex;
  gap: 0.5rem;
  align-items: center;
  margin-left: auto;
}
.topbar .nav a {
  color: #fff;
  padding: 0.4rem 0.7rem;
  border: 1px solid rgba(255,255,255,0.4);
  border-radius: 4px;
  text-decoration: none;
  font-size: 0.9rem;
  min-height: 36px;
  display: inline-flex;
  align-items: center;
}
.topbar .nav a:hover { background: rgba(255,255,255,0.1); text-decoration: none; }
.topbar .logout { margin: 0; }
.topbar .logout button {
  background: transparent;
  color: #fff;
  border: 1px solid #fff;
  padding: 0.35rem 0.7rem;
  border-radius: 4px;
  cursor: pointer;
  min-height: 36px;
}
.topbar .logout button:hover { background: rgba(255,255,255,0.1); }

.nav-toggle,
.search-toggle {
  display: none;
  background: transparent;
  color: #fff;
  border: 1px solid rgba(255,255,255,0.4);
  border-radius: 4px;
  width: 40px;
  height: 40px;
  font-size: 1.2rem;
  padding: 0;
  cursor: pointer;
  align-items: center;
  justify-content: center;
}
.nav-toggle:hover,
.search-toggle:hover { background: rgba(255,255,255,0.1); }

/* Si l'app n'a pas rempli le block topbar_search, la div .topbar-search
   est vide. On cache à la fois la div ET le bouton loupe (qui la précède
   immédiatement dans le markup). Évite à l'app d'avoir à déclarer
   explicitement "j'ai pas de recherche" — le CSS le détecte tout seul.
   :has() est supporté tous navigateurs majeurs depuis fin 2023. */
.topbar-search:empty { display: none !important; }
.topbar .search-toggle:has(~ .topbar-search:empty) { display: none !important; }

@media (max-width: 720px) {
  .topbar {
    flex-wrap: wrap;
    padding: 0.5rem 0.75rem;
    gap: 0.5rem;
  }
  .topbar .brand { flex: 0 0 auto; font-size: 1rem; }
  .topbar .brand-version { display: none; }
  /* Bouton loupe : visible si l'app a déclaré un block topbar_search.
     Posé juste avant le hamburger (order 1 = .nav-toggle). margin-left:
     auto sur le premier des deux pour pousser le groupe à droite. */
  .search-toggle {
    display: inline-flex;
    order: 1;
    margin-left: auto;
  }
  .nav-toggle {
    display: inline-flex;
    order: 1;
    /* v1.8.1 fix : par défaut le hamburger pousse à droite tout seul.
       Quand la loupe est PRÉSENTE ET VISIBLE (topbar_search non vide),
       c'est elle qui prend ce rôle — règle :has() ci-dessous.
       Avant v1.8.1, le sélecteur `.search-toggle ~ .nav-toggle` matchait
       même quand la loupe était `display:none` (combinateurs CSS ignorent
       la visibilité) → hamburger collé au brand sans la recherche. */
    margin-left: auto;
  }
  .topbar:has(.topbar-search:not(:empty)) .nav-toggle { margin-left: 0; }
  /* La barre de recherche en mobile : pleine largeur sous la topbar,
     révélée par toggle. Order 50 = entre toggles (1) et nav (100). */
  .topbar-search {
    order: 50;
    flex-basis: 100%;
    display: none;
    padding-top: 0.25rem;
    border-top: 1px solid rgba(255,255,255,0.15);
    margin-left: 0;
  }
  .topbar-search.is-open { display: block; }
  /* Inputs de la search-form prennent toute la largeur en mobile. */
  .topbar-search form { width: 100%; }
  .topbar-search input[type="search"],
  .topbar-search input[type="text"] { width: 100%; }
  .topbar .nav {
    order: 100;
    flex-basis: 100%;
    flex-direction: column;
    align-items: stretch;
    gap: 0.35rem;
    display: none;
    padding-top: 0.25rem;
    border-top: 1px solid rgba(255,255,255,0.15);
    margin-left: 0;
  }
  .topbar .nav.is-open { display: flex; }
  .topbar .nav a,
  .topbar .nav .logout button { width: 100%; text-align: center; justify-content: center; }
}

/* ─── Buttons / messages ────────────────────────────────────────── */
button {
  padding: 0.45rem 0.9rem;
  background: #1f2937;
  color: #fff;
  border: 0;
  border-radius: 4px;
  cursor: pointer;
  font-size: 0.95rem;
}
button:hover { background: #111827; }
button.danger { background: #b91c1c; }
button.danger:hover { background: #991b1b; }

.error { color: #b91c1c; }
.ok { color: #065f46; }
.meta { color: var(--fg-muted); font-size: 0.9rem; }

/* ─── Tables ────────────────────────────────────────────────────── */
table { width: 100%; border-collapse: collapse; background: var(--card-bg); border-radius: 6px; overflow: hidden; }
table th, table td {
  padding: 0.55rem 0.75rem;
  border-bottom: 1px solid var(--border);
  text-align: left;
  font-size: 0.95rem;
}
table .nowrap, td.nowrap, th.nowrap { white-space: nowrap; }
table th { background: var(--surface-2); font-weight: 600; }
table td.empty { text-align: center; color: #888; padding: 1.5rem; }

.table-wrap { overflow-x: auto; -webkit-overflow-scrolling: touch; }
/* Opt-in : ajoute `.table-wrap--wide` quand la table contient beaucoup
   de colonnes et a besoin d'être large même au prix d'un scroll horizontal
   sur mobile. Par défaut on laisse le contenu décider — la plupart des
   tables /settings/* (3-4 colonnes) tiennent en 320px sans forcer 720px. */
.table-wrap--wide table { min-width: 720px; }

/* ─── Login ─────────────────────────────────────────────────────── */
.login {
  max-width: 360px;
  margin: 4rem auto;
  background: var(--card-bg);
  padding: 1.5rem;
  border-radius: 6px;
  border: 1px solid var(--border);
}
.login form { display: flex; flex-direction: column; gap: 0.75rem; }
.login label { display: flex; flex-direction: column; gap: 0.25rem; font-size: 0.9rem; color: #555; }
.login input { padding: 0.5rem; border: 1px solid #ccc; border-radius: 4px; font-size: 1rem; }

/* ─── Welcome stub ─────────────────────────────────────────────── */
.welcome {
  background: var(--card-bg);
  padding: 2rem 1.5rem;
  border-radius: 6px;
  border: 1px solid var(--border);
  text-align: center;
}
.welcome code {
  background: var(--surface-2);
  padding: 0.1rem 0.3rem;
  border-radius: 3px;
  font-size: 0.85rem;
}


/* ─── Search form (generic skeleton, apps may extend) ───────────── */
.search-form {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 0.5rem;
  margin-bottom: 1rem;
}
.search-form input[type="search"] {
  flex: 1 1 240px;
  min-width: 0;
  padding: 0.5rem 0.75rem;
  border: 1px solid #d1d5db;
  border-radius: 4px;
}
.search-form select {
  padding: 0.5rem;
  border: 1px solid #d1d5db;
  border-radius: 4px;
  background: var(--card-bg);
}
.search-form > button[type="submit"] { flex: 0 0 auto; }

/* ─── Pagination (used by _macros.pagination) ───────────────────── */
.pagination {
  display: flex;
  flex-wrap: wrap;
  gap: 0.3rem;
  align-items: center;
  justify-content: center;
  padding: 1rem 0 0.25rem;
  margin-top: 1rem;
  border-top: 1px solid var(--border);
  font-size: 0.95rem;
}
.pagination a, .pagination span {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  min-width: 40px;
  min-height: 40px;
  padding: 0 0.65rem;
  border-radius: 4px;
  text-decoration: none;
  font-variant-numeric: tabular-nums;
  border: 1px solid transparent;
}
.pagination a { color: var(--color-primary); border-color: #e5e7eb; background: var(--card-bg); }
.pagination a:hover { background: var(--color-primary-bg); text-decoration: none; border-color: #93c5fd; }
.pagination .current {
  background: var(--color-primary);
  color: #fff;
  border-color: var(--color-primary);
  font-weight: 600;
}
.pagination .disabled { color: #9ca3af; cursor: not-allowed; background: transparent; }
.pagination .page-ellipsis { color: #9ca3af; min-width: 24px; padding: 0 0.25rem; border: 0; }
.pagination .page-step { font-weight: 500; gap: 0.25rem; }
@media (max-width: 480px) {
  .pagination { gap: 0.2rem; padding-top: 0.75rem; }
  .pagination a, .pagination span { min-width: 36px; min-height: 40px; padding: 0 0.5rem; font-size: 0.9rem; }
  .pagination .page-step-label {
    position: absolute;
    width: 1px; height: 1px;
    padding: 0; margin: -1px;
    overflow: hidden;
    clip: rect(0,0,0,0);
    white-space: nowrap;
    border: 0;
  }
}

/* ─── Select-all bar + bulk-bar (used by bulk.js) ──────────────── */
.select-all-bar {
  display: flex;
  align-items: center;
  padding: 0.4rem 0.6rem;
  margin: 0.25rem 0 0.5rem;
  background: var(--surface-3);
  border: 1px dashed var(--border);
  border-radius: 4px;
  font-size: 0.85rem;
  color: #374151;
}
.select-all-bar label {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  cursor: pointer;
  user-select: none;
}
.select-all-bar input[type="checkbox"] { width: 18px; height: 18px; cursor: pointer; }

.bulk-bar {
  display: flex;
  gap: 0.5rem;
  align-items: center;
  padding: 0.6rem 0.9rem;
  margin: 0.5rem 0 0.75rem;
  background: var(--color-primary-bg);
  border: 1px solid #bfdbfe;
  border-radius: 6px;
  font-size: 0.95rem;
}
.bulk-bar button { padding: 0.3rem 0.75rem; }

/* ─── View switcher (used by _macros.view_switcher + ui.js) ─────── */
.view-switcher {
  display: inline-flex;
  background: var(--surface-2);
  border: 1px solid var(--border);
  border-radius: 6px;
  padding: 2px;
  gap: 2px;
}
.view-switcher button {
  background: transparent;
  color: #374151;
  border: 0;
  padding: 0.4rem 0.7rem;
  border-radius: 4px;
  font-size: 0.85rem;
  cursor: pointer;
  min-height: 36px;
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
}
.view-switcher button:hover { background: #e5e7eb; }
.view-switcher button.is-active {
  background: var(--card-bg);
  color: var(--color-primary);
  box-shadow: 0 1px 2px rgba(0,0,0,0.06);
  font-weight: 600;
}

/* ─── Settings two-column layout (left nav + content) ───────────── */
.settings-layout {
  display: grid;
  grid-template-columns: 200px minmax(0, 1fr);
  gap: 1.5rem;
  align-items: start;
}
.settings-nav {
  position: sticky;
  top: 1rem;
  display: flex;
  flex-direction: column;
  gap: 0.75rem;
}
.settings-nav ul { list-style: none; margin: 0; padding: 0; display: flex; flex-direction: column; gap: 0.15rem; }
.settings-nav a {
  display: block;
  padding: 0.5rem 0.75rem;
  border-radius: 4px;
  color: #1f2937;
  text-decoration: none;
  font-size: 0.95rem;
}
.settings-nav a:hover { background: var(--surface-2); text-decoration: none; }
.settings-nav a.is-active { background: #1f2937; color: #fff; }

/* Chaque groupe (Framework / Application) dans son propre cadre.
   Couleur d'accent dérivée de l'origine pour rappeler la distinction
   déjà présente sur la page index settings (cf. v0.9.3 / v0.9.5). */
.settings-nav-group {
  background: var(--card-bg);
  border: 1px solid var(--border);
  border-top: 3px solid var(--border);
  border-radius: 6px;
  padding: 0.5rem;
}
.settings-nav-group-framework { border-top-color: #64748b; }
.settings-nav-group-app { border-top-color: var(--color-primary, #2563eb); }
.settings-nav-group-header {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  opacity: 0.6;
  padding: 0.25rem 0.5rem 0.5rem 0.5rem;
  font-weight: 700;
  user-select: none;
  border-bottom: 1px solid var(--border);
  margin-bottom: 0.4rem;
}
.settings-content {
  display: flex;
  flex-direction: column;
  gap: 1.25rem;
  min-width: 0;
}
.settings-section h2 { margin-top: 0; }
.settings-block {
  background: var(--card-bg);
  padding: 1rem 1.25rem;
  border: 1px solid var(--border);
  border-radius: 6px;
  margin-bottom: 1rem;
  overflow-x: auto;
  -webkit-overflow-scrolling: touch;
}
.settings-block h3 { margin-top: 0; font-size: 1rem; color: #111827; }

/* Cadres de regroupement de la page index settings : visuel distinct
   entre les sections du Framework (socle) et celles de l'Application
   (blueprints métier). Liseré gauche + emoji devant le titre. */
.settings-group { border-left: 3px solid var(--border); }
.settings-group-framework { border-left-color: #64748b; }
.settings-group-app { border-left-color: var(--color-primary, #2563eb); }
.settings-group h3 {
  display: flex;
  align-items: center;
  gap: 0.4rem;
  margin-bottom: 0.75rem;
}
.settings-group-framework h3::before { content: "🛠"; }
.settings-group-app h3::before { content: "📦"; }

@media (max-width: 720px) {
  /* `minmax(0, 1fr)` (et non `1fr` seul) est CRITIQUE : sans le min:0,
     les enfants grid héritent de `min-width: auto` = largeur de contenu,
     ce qui fait déborder la sidebar (row scrollable) en dehors du viewport
     et entraîne un scroll horizontal global. Fix v0.13.4. */
  .settings-layout { grid-template-columns: minmax(0, 1fr); gap: 0.75rem; }
  .settings-nav {
    position: static;
    gap: 0.5rem;
    /* min-width: 0 sur les enfants flex/grid pour permettre le shrink
       sous leur largeur de contenu (sans ça, le <ul> en row pousse
       .settings-nav au-delà du viewport et toute la page scroll). */
    min-width: 0;
    max-width: 100%;
  }
  .settings-nav-group {
    padding: 0.35rem 0.5rem 0.5rem 0.5rem;
    overflow-x: auto;
    -webkit-overflow-scrolling: touch;
    /* Idem : sans min-width: 0, le cadre prend la largeur de son contenu
       (le <ul> row) et déborde du viewport au lieu de scroller en
       interne. C'est le bug le plus tordu de flex/grid sur mobile. */
    min-width: 0;
    max-width: 100%;
  }
  .settings-nav-group-header {
    padding: 0.15rem 0.25rem 0.35rem 0.25rem;
    margin-bottom: 0.3rem;
  }
  .settings-nav ul {
    flex-direction: row;
    flex-wrap: nowrap;
    gap: 0.25rem;
  }
  .settings-nav a {
    white-space: nowrap;
    padding: 0.4rem 0.65rem;
    font-size: 0.9rem;
  }
}

/* ─── Generic form helpers (stacked, form-row, inline) ──────────── */
.stacked-form { display: flex; flex-direction: column; gap: 0.75rem; max-width: 540px; }
.stacked-form label { display: flex; flex-direction: column; gap: 0.25rem; font-size: 0.9rem; color: #374151; }
.stacked-form input,
.stacked-form select,
.stacked-form textarea {
  padding: 0.45rem 0.6rem;
  border: 1px solid #d1d5db;
  border-radius: 4px;
  font: inherit;
}
.stacked-form .hint { color: var(--fg-muted); font-size: 0.8rem; }
.form-row { display: flex; gap: 0.75rem; flex-wrap: wrap; align-items: end; }
.form-row label { flex: 1; min-width: 140px; }
.form-checkbox { flex-direction: row !important; align-items: center; gap: 0.5rem; }
.form-checkbox input { margin: 0; }

/* ─── Progress banner (used by polling.js / attachProgressBanner) ── */
/* Sticky en haut, sous la topbar, pour les jobs longs avec UI de
   feedback. Le markup attendu :
     <div class="progress-banner" hidden>
       <span data-banner-message>…</span>
       <div class="progress-bar"><span data-banner-bar></span></div>
     </div>
   `hidden` est géré par polling.js — pas besoin de le toucher en CSS. */
.progress-banner {
  background: var(--color-primary-bg);
  color: var(--color-primary-dark);
  border: 1px solid var(--color-primary-border);
  border-radius: 6px;
  padding: 0.6rem 0.9rem;
  margin: 0.75rem auto;
  max-width: 980px;
  display: flex;
  align-items: center;
  gap: 0.75rem;
  font-size: 0.9rem;
}
.progress-banner [data-banner-message] { flex: 0 1 auto; }
.progress-banner .progress-bar {
  flex: 1;
  height: 8px;
  background: rgba(255,255,255,0.5);
  border-radius: 4px;
  overflow: hidden;
}
.progress-banner [data-banner-bar] {
  display: block;
  height: 100%;
  width: 0%;
  background: var(--color-primary);
  transition: width 0.4s ease;
}
:root[data-scheme="dark"] .progress-banner {
  background: var(--surface-2);
  color: var(--fg);
  border-color: var(--border);
}
:root[data-scheme="dark"] .progress-banner .progress-bar {
  background: rgba(255,255,255,0.1);
}

/* ─── Toasts (used by toast.js, #toast-root in base.html) ───────── */
#toast-root {
  position: fixed;
  right: 1rem;
  bottom: 1rem;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  z-index: 1000;
  max-width: calc(100vw - 2rem);
  pointer-events: none;
}
.toast {
  pointer-events: auto;
  min-width: 260px;
  max-width: 360px;
  background: #1f2937;
  color: #f9fafb;
  border-radius: 6px;
  padding: 0.7rem 0.9rem;
  display: flex;
  align-items: flex-start;
  gap: 0.5rem;
  box-shadow: 0 6px 18px rgba(0,0,0,0.18);
  transform: translateY(20px);
  opacity: 0;
  transition: transform 0.2s ease, opacity 0.2s ease;
}
.toast.toast-in { transform: translateY(0); opacity: 1; }
.toast.toast-leaving { opacity: 0; transform: translateY(20px); }
.toast .toast-body {
  flex: 1; min-width: 0;
  display: flex; flex-direction: column;
  gap: 0.25rem; font-size: 0.9rem; line-height: 1.35;
}
.toast .toast-body strong { font-size: 0.95rem; }
.toast .toast-close {
  background: transparent;
  border: 0;
  color: #d1d5db;
  font-size: 1.2rem;
  line-height: 1;
  cursor: pointer;
  padding: 0 0.25rem;
}
.toast .toast-close:hover { color: #fff; background: transparent; }
.toast.toast-success { background: #065f46; }
.toast.toast-error { background: #991b1b; }
.toast.toast-progress { background: #1f2937; }
.toast.toast-info { background: #1f2937; }

@media (max-width: 480px) {
  #toast-root { left: 0.5rem; right: 0.5rem; bottom: 0.5rem; max-width: none; }
  .toast { min-width: 0; max-width: none; }
}

/* ─── Confirm modal (used by confirm.js, [data-confirm]) ────────── */
#confirm-overlay {
  position: fixed;
  inset: 0;
  background: rgba(17, 24, 39, 0.55);
  display: flex;
  align-items: center;
  justify-content: center;
  padding: 1rem;
  z-index: 1100;
  opacity: 0;
  transition: opacity 0.15s ease;
}
#confirm-overlay.is-open { opacity: 1; }
#confirm-overlay[hidden] { display: none; }
.confirm-modal {
  background: var(--card-bg);
  border-radius: 8px;
  max-width: 460px;
  width: 100%;
  padding: 1.25rem 1.25rem 1rem;
  box-shadow: 0 18px 50px rgba(0,0,0,0.25);
}
.confirm-modal h2 { margin: 0 0 0.5rem; font-size: 1.05rem; color: #111827; }
.confirm-modal p { margin: 0 0 1rem; color: #374151; font-size: 0.95rem; line-height: 1.4; }
.confirm-actions { display: flex; gap: 0.5rem; justify-content: flex-end; }
.confirm-actions button { min-height: 44px; }
.confirm-cancel { background: #e5e7eb; color: #1f2937; }
.confirm-cancel:hover { background: #d1d5db; }
@media (max-width: 480px) {
  .confirm-actions { flex-direction: column-reverse; }
  .confirm-actions button { width: 100%; }
}

/* ─── Mobile tweaks shared across forms ─────────────────────────── */
/* Baseline cible : iPhone SE 1ʳᵉ gén (320px) à smartphone récent (400px).
   Tous les fix mobile sont consolidés ici en une seule media query pour
   être faciles à auditer. WCAG : touch targets ≥ 44×44px. iOS : font-size
   ≥ 16px sur les inputs (sinon zoom auto au focus — déjà couvert par la
   règle globale `input { font-size: max(16px, 1em); }`). */
@media (max-width: 480px) {
  main { margin: 0.75rem auto; padding: 0 0.6rem; }
  .search-form input[type="search"] { flex-basis: 100%; min-height: 44px; }
  .search-form select { flex: 1; min-height: 44px; }
  .search-form button { min-height: 44px; }
  .bulk-bar { flex-wrap: wrap; }
  .bulk-bar button { flex: 1 1 auto; min-height: 40px; }
  .view-switcher { align-self: flex-end; }

  /* Login : marge top excessive de 4rem inutile en mobile, on serre. */
  .login {
    margin: 1.5rem auto;
    padding: 1.25rem;
  }

  /* Welcome stub : 2rem de padding gaspille l'espace. */
  .welcome { padding: 1.25rem 1rem; }

  /* Tables : la classe `.table-wrap--wide` force 720px pour les tables
     très larges, mais la version par défaut s'adapte au viewport. */

  /* Boutons par défaut : touch target ≥ 44px de hauteur. v1.10.0 : couvre
     aussi .btn-icon (qui mesure 40px par défaut sur desktop) — sinon une
     barre d'icônes en mobile n'atteint pas le seuil WCAG. */
  button { min-height: 44px; }
  .btn-icon { width: 44px; min-width: 44px; height: 44px; min-height: 44px; }

  /* Pagination : conserve 44px de hauteur minimum (override v0.3.0
     qui descendait à 36px — sous le seuil WCAG). */
  .pagination a, .pagination span { min-width: 44px; min-height: 44px; }

  /* Form-row : stack en colonne pour ne pas coller 2 boutons à 130px
     côte-à-côte (cas /settings/integrations Enregistrer + Tester). */
  .form-row { flex-direction: column; align-items: stretch; }
  .form-row label, .form-row button { width: 100%; }

  /* Toast : remonté à 1rem du bas pour ne pas chevaucher les boutons
     d'actions en bas de formulaire. */
  #toast-root { bottom: 1rem; left: 0.5rem; right: 0.5rem; }

  /* Confirm modal : padding réduit, marges latérales 0.5rem pour
     respirer dans 320px. */
  .confirm-modal { padding: 1rem; margin: 0 0.5rem; }

  /* Densité compacte désactivée en mobile : les touch targets exigent
     du padding même si l'admin a coché "compact" en desktop. */
  :root[data-density="compact"] {
    --pad-y: 0.55rem;
    --pad-x: 0.75rem;
    --gap: 0.5rem;
  }
}

/* ─── v0.3.0 : topbar icon buttons, search slot, nav links ──────── */
.topbar .icon-btn {
  background: transparent;
  color: #fff;
  border: 1px solid rgba(255,255,255,0.4);
  border-radius: 4px;
  width: 40px;
  height: 36px;
  padding: 0;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  cursor: pointer;
  text-decoration: none;
}
.topbar .icon-btn:hover { background: rgba(255,255,255,0.1); }
.topbar .logout { margin: 0; }

/* Search bar sitting between brand and nav. Auto-hidden when the app didn't
   fill the `topbar_search` block (the :empty rule below catches that). */
.topbar-search {
  flex: 1 1 auto;
  min-width: 0;
  max-width: 360px;
  margin-left: 0.5rem;
}
.topbar-search:empty { display: none; }
.topbar-search-form { margin: 0; }
.topbar-search-form input[type="search"] {
  width: 100%;
  padding: 0.4rem 0.7rem;
  border-radius: 4px;
  border: 1px solid rgba(255,255,255,0.25);
  background: rgba(255,255,255,0.1);
  color: #fff;
  font-size: 0.9rem;
  min-height: 36px;
}
.topbar-search-form input[type="search"]::placeholder { color: rgba(255,255,255,0.6); }
.topbar-search-form input[type="search"]:focus {
  outline: none;
  border-color: rgba(255,255,255,0.6);
  background: rgba(255,255,255,0.18);
}

/* App's top-level navigation links (rendered by _macros.nav_link). */
.topbar .nav-link {
  color: #fff;
  padding: 0.4rem 0.7rem;
  border-radius: 4px;
  text-decoration: none;
  font-size: 0.9rem;
  min-height: 36px;
  display: inline-flex;
  align-items: center;
  border: 1px solid transparent;
}
.topbar .nav-link:hover { background: rgba(255,255,255,0.1); text-decoration: none; }
.topbar .nav-link.is-active {
  background: rgba(255,255,255,0.15);
  border-color: rgba(255,255,255,0.3);
}

@media (max-width: 720px) {
  .topbar-search { order: 50; flex-basis: 100%; max-width: none; margin: 0; display: none; }
  .topbar .nav.is-open .topbar-search,
  .topbar.is-mobile-open .topbar-search { display: block; }
}

/* ─── v0.3.0 : theme settings page (swatches, options) ──────────── */
/* v1.10.0 : la page Apparence a un form dense (6+ fieldsets, swatches,
   pipettes, sliders) qui s'étouffe avec le max-width: 540px posé sur
   .stacked-form (utile pour Compte avec ses inputs étroits, pénalisant
   ici). Override ciblé : si le form contient des .theme-swatches, c'est
   forcément le form Apparence — on libère la largeur. */
.stacked-form:has(.theme-swatches) { max-width: none; }
.theme-swatches { display: flex; flex-wrap: wrap; gap: 0.75rem; padding: 0; }
.theme-swatch {
  display: inline-flex;
  flex-direction: column;
  align-items: center;
  gap: 0.3rem;
  cursor: pointer;
  font-size: 0.85rem;
  padding: 0.5rem;
  border-radius: 6px;
  border: 2px solid transparent;
}
.theme-swatch input[type="radio"] { position: absolute; opacity: 0; pointer-events: none; }
.theme-swatch:has(input:checked) { border-color: var(--fg); background: var(--surface-2); }
.swatch-dot {
  width: 32px;
  height: 32px;
  border-radius: 50%;
  border: 2px solid var(--border-strong);
}
.theme-swatch-blue   .swatch-dot { background: #1d4ed8; }
.theme-swatch-green  .swatch-dot { background: #059669; }
.theme-swatch-violet .swatch-dot { background: #7c3aed; }
.theme-swatch-orange .swatch-dot { background: #ea580c; }
.theme-swatch-teal   .swatch-dot { background: #0d9488; }
.theme-swatch-slate  .swatch-dot { background: #475569; }

.theme-options { display: flex; flex-wrap: wrap; gap: 0.75rem; }
.theme-options label {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.4rem 0.75rem;
  border: 1px solid var(--border);
  border-radius: 4px;
  cursor: pointer;
  background: var(--surface-3);
}
.theme-options label:has(input:checked) {
  background: var(--color-primary-bg);
  border-color: var(--color-primary);
  /* Light mode : texte teinté foncé sur fond pâle. Contraste OK. */
  color: var(--color-primary-dark);
  font-weight: 600;
}

/* Font-option previews — "Aa" rendered in each font so the user picks by
   sight. Class names match the cookie value (theme-font-{system|sans|serif|mono}). */
.theme-font-option .font-sample {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 32px;
  height: 32px;
  border-radius: 4px;
  background: var(--card-bg);
  border: 1px solid var(--border);
  font-weight: 600;
  font-size: 1.05rem;
  color: var(--fg);
}
.theme-font-system .font-sample { font-family: system-ui, -apple-system, "Segoe UI", Roboto, sans-serif; }
.theme-font-sans   .font-sample { font-family: Arial, Helvetica, sans-serif; }
.theme-font-serif  .font-sample { font-family: Georgia, "Times New Roman", Times, serif; }
.theme-font-mono   .font-sample { font-family: "SF Mono", Menlo, Consolas, "Courier New", monospace; }

fieldset {
  border: 1px solid var(--border);
  border-radius: 6px;
  padding: 0.75rem 1rem 1rem;
  margin: 0.5rem 0;
}
fieldset legend { padding: 0 0.4rem; font-weight: 600; color: var(--fg); }

/* ─── v0.3.0 : compact density tweaks ───────────────────────────── */
:root[data-density="compact"] table th,
:root[data-density="compact"] table td { padding: 0.35rem 0.5rem; font-size: 0.9rem; }
:root[data-density="compact"] .settings-block { padding: 0.6rem 0.8rem; }
:root[data-density="compact"] main { margin: 0.75rem auto; }

/* ─── v0.3.0 : dark scheme adjustments for components with dark
   colors baked in (topbar already is, login/etc inherit via tokens) ─ */
:root[data-scheme="dark"] .login,
:root[data-scheme="dark"] .welcome { box-shadow: 0 2px 12px rgba(0,0,0,0.4); }
:root[data-scheme="dark"] .pagination a { background: var(--card-bg); color: #93c5fd; }
:root[data-scheme="dark"] .pagination a:hover { background: var(--surface-3); }
:root[data-scheme="dark"] table th { color: var(--fg); }
:root[data-scheme="dark"] .confirm-modal { color: var(--fg); }
:root[data-scheme="dark"] .confirm-modal p,
:root[data-scheme="dark"] .confirm-modal h2 { color: var(--fg); }

/* ─── v0.5.8 : reste des couleurs hardcodees a forcer en dark mode ─
   Beaucoup de regles plus haut utilisent des couleurs sombres en dur
   (#111827, #1f2937, #374151, #555). En light mode c'est OK, en dark
   c'est du texte sombre sur fond sombre = illisible. On les override
   ici pour pointer sur var(--fg) (ou un equivalent muted) en dark. */
:root[data-scheme="dark"] h1,
:root[data-scheme="dark"] h2,
:root[data-scheme="dark"] .settings-block h3 { color: var(--fg); }

:root[data-scheme="dark"] .login label,
:root[data-scheme="dark"] .stacked-form label,
:root[data-scheme="dark"] .form-row > label,
:root[data-scheme="dark"] .view-switcher button,
:root[data-scheme="dark"] .select-all-bar { color: var(--fg); }

:root[data-scheme="dark"] .settings-nav a { color: var(--fg); }
:root[data-scheme="dark"] .settings-nav a:hover { background: var(--surface-3); }
:root[data-scheme="dark"] .settings-nav a.is-active { background: var(--color-primary-dark); color: #fff; }

/* /settings/theme : en dark mode, l'option sélectionnée bascule en fond
   teinté foncé + texte blanc. La règle light (texte foncé sur fond pâle
   `--color-primary-bg`) ne marche pas en dark parce que `theme-boot.js`
   pose toujours le `--color-primary-bg` clair (pas d'alternative dark
   dynamique) — le texte serait foncé sur pâle, mais visuellement
   inconfortable et trop discret en dark. Inverser le couple donne le
   meilleur contraste possible. */
:root[data-scheme="dark"] .theme-options label:has(input:checked) {
  background: var(--color-primary-dark);
  border-color: var(--color-primary);
  color: #fff;
}

/* Champs de formulaire : background neutre, bordure visible, texte clair */
:root[data-scheme="dark"] .login input,
:root[data-scheme="dark"] .stacked-form input,
:root[data-scheme="dark"] .stacked-form select,
:root[data-scheme="dark"] .stacked-form textarea {
  background: var(--surface-3);
  border-color: var(--border-strong);
  color: var(--fg);
}
:root[data-scheme="dark"] .stacked-form input:focus,
:root[data-scheme="dark"] .stacked-form select:focus,
:root[data-scheme="dark"] .stacked-form textarea:focus {
  outline-color: var(--color-primary);
}

/* Status colors : versions plus lumineuses pour passer sur fond sombre */
:root[data-scheme="dark"] .ok { color: #34d399; }
:root[data-scheme="dark"] .error { color: #f87171; }
:root[data-scheme="dark"] table td.empty { color: var(--fg-muted); }

/* ─── v1.3.0 : dark contrast — texte des badges/elements qui utilisent
   --color-primary-dark comme couleur sur fond --color-primary-bg.
   En dark mode, theme-boot remplace --color-primary-bg par bg_dark
   (rgba semi-transparent du primary, lightness 55%). La couleur de texte
   par défaut (--color-primary-dark, lightness 15-30%) devient invisible
   sur ce fond foncé semi-transparent posé sur la surface dark
   (--surface-2 = #1e293b). On bascule le texte vers --color-primary-border
   (HSL ~75% lightness) qui suit le skin et reste lisible. */
:root[data-scheme="dark"] .badge-primary { color: var(--color-primary-border); }
/* v1.10.0 : le fallback dark `--color-primary-bg` (anciennement hardcodé en
   bleu ici) est désormais géré déclarativement par preset, cf. section
   "Color presets" en tête. Plus de dépendance JS pour le contraste sombre. */

/* ─── v0.5.9 : /settings/system metrics table ───────────────────── */
.metrics-table { width: 100%; }
.metrics-table th {
  background: transparent;
  width: 12rem;
  vertical-align: top;
  padding-top: 0.7rem;
}
.metrics-table td { padding: 0.5rem 0.75rem 0.7rem; }
.metric-value { font-size: 1.05rem; font-weight: 600; }
.metric-detail { font-size: 0.85rem; color: var(--fg-muted); margin-top: 0.15rem; }
.metric-bar {
  width: 100%;
  max-width: 320px;
  height: 8px;
  background: var(--surface-2);
  border-radius: 4px;
  overflow: hidden;
  margin-top: 0.4rem;
}
.metric-bar-fill {
  height: 100%;
  background: var(--color-primary);
  border-radius: 4px;
  transition: width 0.3s ease;
}
.metric-bar-fill.warn { background: #f59e0b; }
.metric-bar-fill.danger { background: #dc2626; }
.metric-percent { font-size: 0.8rem; color: var(--fg-muted); margin-top: 0.15rem; }

/* ─── v0.8.0 : custom primary color swatch + background image ────── */
/* "Personnalisée" swatch : un dégradé arc-en-ciel signale qu'on peut
   choisir n'importe quelle couleur via la pipette native. */
.theme-swatch-custom .swatch-dot {
  background: conic-gradient(from 0deg,
    #ef4444, #f59e0b, #eab308, #22c55e,
    #06b6d4, #3b82f6, #8b5cf6, #ec4899, #ef4444);
}
.theme-color-custom {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  margin-top: 0.8rem;
  padding-top: 0.7rem;
  border-top: 1px dashed var(--border);
  font-size: 0.9rem;
  color: var(--fg-muted);
}
.theme-color-custom input[type="color"] {
  width: 48px;
  height: 32px;
  padding: 0;
  border: 1px solid var(--border-strong);
  border-radius: 4px;
  cursor: pointer;
  background: var(--card-bg);
}

/* v1.5.0 : sous-bloc d'un fieldset (couleur unie / degrade) reutilise
   le styling du .theme-color-custom mais autorise plusieurs inputs cote
   a cote (2 color pickers + 1 select pour le degrade). */
.theme-bg-suboption {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  margin-top: 0.6rem;
  padding-top: 0.5rem;
  border-top: 1px dashed var(--border);
  font-size: 0.9rem;
  color: var(--fg-muted);
  flex-wrap: wrap;
}
.theme-bg-suboption input[type="color"] {
  width: 48px;
  height: 32px;
  padding: 0;
  border: 1px solid var(--border-strong);
  border-radius: 4px;
  cursor: pointer;
  background: var(--card-bg);
}
.theme-bg-suboption select { max-width: 260px; }

/* Background custom (v1.5.0+) — applique via theme-boot.js qui pose :
   - body[data-bg="on"] : un fond custom est actif (vs theme par defaut)
   - body[data-bg-mode]  : type du fond (image | color | gradient)
   - :root --bg-image-url, --bg-overlay-opacity (mode image)
   - :root --bg-custom (mode color ou gradient — soit un hex, soit une
     linear-gradient(...) prete a poser sur background) */

/* Mode image : voile + image (existant pre-v1.5.0). Le voile utilise
   color-mix() pour conserver la teinte du theme (dark/light) tout en
   restant translucide. Modern browsers : Chrome 111+, Firefox 113+,
   Safari 16.2+. */
body[data-bg="on"][data-bg-mode="image"] {
  background-image:
    linear-gradient(
      color-mix(in srgb, var(--bg) calc(var(--bg-overlay-opacity, 0.5) * 100%), transparent),
      color-mix(in srgb, var(--bg) calc(var(--bg-overlay-opacity, 0.5) * 100%), transparent)
    ),
    var(--bg-image-url);
  background-size: auto, cover;
  background-position: center, center;
  background-attachment: fixed, fixed;
  background-repeat: repeat, no-repeat;
}
/* Mode couleur : fond uni. Le var --bg-custom contient le hex. */
body[data-bg="on"][data-bg-mode="color"] {
  background: var(--bg-custom);
}
/* Mode degrade : le var --bg-custom contient deja le linear-gradient(...)
   genere par theme-boot.js depuis les data-bg-grad-* attributes. fixed
   pour que le degrade reste en place quand on scroll (plus joli sur
   les pages longues). */
body[data-bg="on"][data-bg-mode="gradient"] {
  background: var(--bg-custom) fixed;
  background-attachment: fixed;
}

/* iOS Safari ignore `background-attachment: fixed` (perf reasons). Fallback
   en scroll sur mobile — image et degrade restent visibles, juste pas parallax.
   v1.10.0 : aligné sur breakpoint canonique 720 (était 640 avant). */
@media (max-width: 720px) {
  body[data-bg="on"][data-bg-mode="image"] {
    background-attachment: scroll, scroll;
  }
  body[data-bg="on"][data-bg-mode="gradient"] {
    background-attachment: scroll;
  }
}

/* Preview + slider + actions de l'image de fond dans /settings/theme */
.theme-bg-preview {
  display: block;
  max-width: 100%;
  width: 320px;
  height: 160px;
  margin-bottom: 0.6rem;
  border-radius: 6px;
  border: 1px solid var(--border);
  background-color: var(--surface-3);
  background-size: cover;
  background-position: center;
  background-repeat: no-repeat;
}
.theme-bg-preview.is-empty {
  display: flex;
  align-items: center;
  justify-content: center;
  color: var(--fg-muted);
  font-size: 0.9rem;
}
/* Aperçu favicon (carré, sur damier neutre pour juger la transparence). */
.favicon-preview {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 72px;
  height: 72px;
  margin-bottom: 0.6rem;
  padding: 6px;
  border-radius: 6px;
  border: 1px solid var(--border);
  background-color: var(--surface-3);
}
.favicon-preview img {
  max-width: 100%;
  max-height: 100%;
  object-fit: contain;
}
.theme-bg-actions {
  display: flex;
  flex-wrap: wrap;
  gap: 0.6rem;
  align-items: center;
}
.theme-bg-actions form {
  display: inline-flex;
  gap: 0.4rem;
  align-items: center;
  margin: 0;
}
.theme-opacity {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  margin-top: 0.6rem;
}
.theme-opacity input[type="range"] {
  flex: 1;
  max-width: 260px;
}
.theme-opacity output {
  font-variant-numeric: tabular-nums;
  min-width: 3ch;
  text-align: right;
  color: var(--fg-muted);
}

/* ═══════════════════════════════════════════════════════════════════
   DESIGN SYSTEM v1.1.0
   ───────────────────────────────────────────────────────────────────
   Classes réutilisables, cohérentes, mobile-friendly. À privilégier
   par rapport aux styles ad-hoc côté app. Conventions :
   - Toutes les couleurs via CSS custom properties (theme-aware)
   - Touch targets ≥ 44px sur tous les éléments interactifs (mobile)
   - Focus visible sur tab navigation (accessibilité)
   ═══════════════════════════════════════════════════════════════════ */

/* ─── Icônes SVG inline (rendues par app/icons.py) ─────────────────── */
.icon {
  display: inline-block;
  vertical-align: -0.15em;
  flex-shrink: 0;
}
.icon + * { margin-left: 0.4em; }
* + .icon { margin-left: 0.4em; }
button > .icon:only-child,
a > .icon:only-child { margin: 0; }

/* ─── Boutons cohérents (.btn et variantes) ─────────────────────────── */
.btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 0.4rem;
  padding: 0.55rem 1rem;
  min-height: 40px;
  border: 1px solid transparent;
  border-radius: var(--radius-md);
  font: inherit;
  font-size: var(--font-size-md);
  font-weight: 500;
  line-height: 1.2;
  cursor: pointer;
  text-decoration: none;
  transition: background-color 0.12s ease, border-color 0.12s ease, transform 0.05s ease;
  user-select: none;
  white-space: nowrap;
}
.btn:focus-visible { outline: none; box-shadow: var(--focus-ring); }
.btn:active { transform: translateY(1px); }
.btn:disabled, .btn[disabled] { opacity: 0.55; cursor: not-allowed; transform: none; }

/* Variants */
.btn-primary {
  background: var(--color-primary);
  color: #fff;
  border-color: var(--color-primary);
}
.btn-primary:hover:not(:disabled) {
  background: var(--color-primary-dark);
  border-color: var(--color-primary-dark);
  text-decoration: none;
}

.btn-secondary {
  background: var(--card-bg);
  color: var(--fg);
  border-color: var(--border-strong);
}
.btn-secondary:hover:not(:disabled) {
  background: var(--surface-2);
  border-color: var(--fg-muted);
  text-decoration: none;
}

.btn-danger {
  background: var(--color-error);
  color: #fff;
  border-color: var(--color-error);
}
.btn-danger:hover:not(:disabled) {
  background: #b91c1c;
  border-color: #b91c1c;
  text-decoration: none;
}

.btn-ghost {
  background: transparent;
  color: var(--fg);
  border-color: transparent;
}
.btn-ghost:hover:not(:disabled) {
  background: var(--surface-2);
  text-decoration: none;
}

.btn-ghost-danger {
  background: transparent;
  color: var(--color-error);
  border-color: transparent;
}
.btn-ghost-danger:hover:not(:disabled) {
  background: var(--color-error-bg);
  text-decoration: none;
}

/* Sizes */
.btn-sm { padding: 0.35rem 0.7rem; min-height: 32px; font-size: var(--font-size-sm); }
.btn-lg { padding: 0.75rem 1.5rem; min-height: 48px; font-size: var(--font-size-lg); }

/* Icon-only buttons (carré, touch target maintenu) */
.btn-icon {
  padding: 0;
  width: 40px;
  min-width: 40px;
  height: 40px;
  min-height: 40px;
}
.btn-icon.btn-sm { width: 32px; min-width: 32px; height: 32px; min-height: 32px; }
.btn-icon.btn-lg { width: 48px; min-width: 48px; height: 48px; min-height: 48px; }

/* Groupe de boutons (espacement cohérent) */
.btn-group {
  display: inline-flex;
  gap: var(--gap-sm);
  flex-wrap: wrap;
}

/* ─── Inputs cohérents (focus ring visible, dark mode auto) ─────────── */
.field {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
}
.field-label {
  font-size: var(--font-size-sm);
  font-weight: 500;
  color: var(--fg);
}
.field-hint {
  font-size: var(--font-size-xs);
  color: var(--fg-muted);
}
.field-error {
  font-size: var(--font-size-xs);
  color: var(--color-error);
}
/* Checkbox + label sur une ligne (case à cocher d'un réglage booléen). */
.field-check {
  display: flex;
  flex-direction: row;
  align-items: center;
  gap: 0.5rem;
  cursor: pointer;
  font-size: var(--font-size-sm);
}
.field-check input[type="checkbox"] {
  width: 18px;
  height: 18px;
  margin: 0;
  flex-shrink: 0;
  cursor: pointer;
}

input[type="text"],
input[type="search"],
input[type="email"],
input[type="password"],
input[type="number"],
input[type="url"],
input[type="tel"],
select,
textarea {
  display: inline-block;
  width: auto;
  padding: 0.5rem 0.7rem;
  border: 1px solid var(--border-strong);
  border-radius: var(--radius-md);
  background: var(--card-bg);
  color: var(--fg);
  font: inherit;
  line-height: 1.4;
  transition: border-color 0.12s ease, box-shadow 0.12s ease;
}
input[type="text"]:focus,
input[type="search"]:focus,
input[type="email"]:focus,
input[type="password"]:focus,
input[type="number"]:focus,
input[type="url"]:focus,
input[type="tel"]:focus,
select:focus,
textarea:focus {
  outline: none;
  border-color: var(--color-primary);
  box-shadow: var(--focus-ring);
}
input:disabled,
select:disabled,
textarea:disabled {
  opacity: 0.6;
  cursor: not-allowed;
  background: var(--surface-2);
}
.input-full { width: 100%; }

/* Field-error state (input avec erreur) */
.input-error {
  border-color: var(--color-error) !important;
  box-shadow: 0 0 0 3px var(--color-error-bg);
}

/* ─── Panel (card avec header / body / footer) ─────────────────────── */
.panel {
  background: var(--card-bg);
  border: 1px solid var(--border);
  border-radius: var(--radius-md);
  margin-bottom: var(--gap-lg);
  overflow: hidden;
}
.panel-elevated { box-shadow: var(--shadow-sm); }
.panel-flat { border: none; background: transparent; }

.panel-header {
  display: flex;
  align-items: center;
  gap: var(--gap-md);
  padding: 0.75rem 1rem;
  border-bottom: 1px solid var(--border);
  background: var(--surface-3);
}
.panel-header h2,
.panel-header h3 {
  margin: 0;
  flex: 1;
  font-size: var(--font-size-lg);
  color: var(--fg);
}
.panel-header h3 { font-size: var(--font-size-md); font-weight: 600; }
.panel-header .panel-count {
  font-size: var(--font-size-sm);
  color: var(--fg-muted);
  background: var(--surface-2);
  padding: 0.15rem 0.5rem;
  border-radius: var(--radius-pill);
  font-variant-numeric: tabular-nums;
}
.panel-header .panel-actions {
  display: flex;
  gap: var(--gap-sm);
  margin-left: auto;
}

.panel-body { padding: 1rem; }
.panel-body > :first-child { margin-top: 0; }
.panel-body > :last-child { margin-bottom: 0; }

.panel-footer {
  padding: 0.75rem 1rem;
  border-top: 1px solid var(--border);
  background: var(--surface-3);
  display: flex;
  gap: var(--gap-sm);
  justify-content: flex-end;
}

.panel-intro {
  padding: 0.75rem 1rem 0;
  color: var(--fg-muted);
  font-size: var(--font-size-sm);
}

/* ─── List-edit (liste de rows éditables avec actions inline) ──────── */
/* Cas typique : config de catégories, mappings clé→valeur, items
   nommés avec un chemin. Voir CLAUDE.md section Design system. */
.list-edit {
  display: flex;
  flex-direction: column;
  gap: var(--gap-sm);
}
.list-edit-row {
  display: flex;
  align-items: center;
  gap: var(--gap);
  padding: 0.5rem 0.75rem;
  background: var(--card-bg);
  border: 1px solid var(--border);
  border-radius: var(--radius-md);
  transition: border-color 0.12s ease, background 0.12s ease;
}
.list-edit-row:hover { border-color: var(--border-strong); }
.list-edit-row .list-edit-icon {
  color: var(--fg-muted);
  flex-shrink: 0;
}
.list-edit-row .list-edit-fields {
  display: flex;
  gap: var(--gap-sm);
  flex: 1;
  min-width: 0;
}
.list-edit-row .list-edit-fields input { flex: 1; min-width: 0; }
.list-edit-row .list-edit-actions {
  display: flex;
  gap: 0.25rem;
  flex-shrink: 0;
}

@media (max-width: 720px) {
  .list-edit-row { flex-wrap: wrap; }
  .list-edit-row .list-edit-fields { flex-basis: 100%; flex-direction: column; }
  .list-edit-row .list-edit-actions { margin-left: auto; }
}

/* ─── État vide chaleureux ─────────────────────────────────────────── */
.empty {
  text-align: center;
  padding: 2.5rem 1rem;
  color: var(--fg-muted);
}
.empty .icon {
  color: var(--border-strong);
  margin-bottom: 0.75rem;
}
.empty-title {
  font-size: var(--font-size-lg);
  font-weight: 600;
  color: var(--fg);
  margin: 0 0 0.4rem 0;
}
.empty-hint {
  font-size: var(--font-size-sm);
  max-width: 360px;
  margin: 0 auto 1.25rem auto;
  line-height: 1.5;
}
.empty-action { margin-top: 0.5rem; }

@media (max-width: 480px) {
  .empty { padding: 1.5rem 0.75rem; }
}

/* ─── Badges (compteurs, statuts) ──────────────────────────────────── */
.badge {
  display: inline-flex;
  align-items: center;
  gap: 0.25rem;
  padding: 0.15rem 0.55rem;
  border-radius: var(--radius-pill);
  background: var(--surface-2);
  color: var(--fg);
  font-size: var(--font-size-xs);
  font-weight: 500;
  font-variant-numeric: tabular-nums;
}
.badge-primary { background: var(--color-primary-bg); color: var(--color-primary-dark); }
.badge-success { background: var(--color-success-bg); color: var(--color-success); }
.badge-warn { background: var(--color-warn-bg); color: var(--color-warn); }
.badge-error { background: var(--color-error-bg); color: var(--color-error); }
.badge-info { background: var(--color-info-bg); color: var(--color-info); }

/* ─── Alert / message (variantes sémantiques) ──────────────────────── */
.alert {
  display: flex;
  gap: 0.6rem;
  padding: 0.75rem 1rem;
  border: 1px solid var(--border);
  border-radius: var(--radius-md);
  background: var(--surface-3);
  color: var(--fg);
  margin-bottom: var(--gap-md);
}
.alert .icon { flex-shrink: 0; margin-top: 0.1em; }
.alert > div { flex: 1; }
.alert-success { background: var(--color-success-bg); border-color: var(--color-success-border); color: var(--color-success); }
.alert-warn { background: var(--color-warn-bg); border-color: var(--color-warn-border); color: var(--color-warn); }
.alert-error { background: var(--color-error-bg); border-color: var(--color-error-border); color: var(--color-error); }
.alert-info { background: var(--color-info-bg); border-color: var(--color-info-border); color: var(--color-info); }

