/* ============================================================================
   odin-base.css — GLOBALE BASIS-KOMPONENTER (D-357, Etape 1 Foundation)
   ============================================================================
   Loades i base.html EFTER app.css. Indeholder:
     1. Global ADDITIV .btn:focus-visible-ring (a11y-retrofit — vises kun ved
        keyboard-fokus, ændrer INTET i hvile → 0 snapshot-regression).
     2. Token-knapsystemet (6 varianter × 3 størrelser + ikon/loading-modifiers)
        scoped under .odin-ui så det IKKE overskriver legacy app.css .btn globalt.
        Opt-in: wrap en container i class="odin-ui" (eller brug button-makroen
        inde i en .odin-ui-scope). Global override af legacy .btn er Wave 2 (gated).
     3. Globalt sticky-nav-mønster (.page-header-sticky / .tab-nav-sticky), løftet
        og tokeniseret fra spaadom/_layout.html's .spaa-sticht (D-355 pilot).

   Forudsætter odin-tokens.css (loadet før app.css).
   ============================================================================ */

/* ── 1. GLOBAL a11y-fokusring (additiv — gælder ALLE .btn, også legacy) ──────
   app.css' .btn har INGEN :focus-style → keyboard-brugere ser intet fokus i dag.
   Denne ring er solid + halo, så den ses på både dark surfaces og accent-flader
   (--border-focus 0.5α-copper er usynlig copper-på-copper). Vises kun ved
   :focus-visible (tastatur), aldrig ved muse-klik → ingen hvile-ændring. */
.btn:focus-visible,
button:focus-visible,
[role="button"]:focus-visible,
a.btn:focus-visible {
  outline: 2px solid var(--focus-ring, #03B898);
  outline-offset: 2px;
  box-shadow: 0 0 0 4px var(--focus-ring-halo, rgba(0, 0, 0, 0.6));
  border-radius: var(--radius-md, 6px);
}

/* ── 2. TOKEN-KNAPSYSTEM (opt-in via .odin-ui) ───────────────────────────────
   6 varianter: primary | secondary | ghost | danger | success + ikon/loading-modifiers.
   .odin-ui .btn-x (0,2,0) vinder over legacy app.css .btn-x (0,1,0) inde i scope. */
.odin-ui .btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: var(--space-2);
  border-radius: var(--radius-md);
  border: 1px solid transparent;
  font-family: inherit;
  font-weight: var(--font-weight-medium);
  line-height: 1;
  cursor: pointer;
  text-decoration: none;
  white-space: nowrap;
  user-select: none;
  transition: background var(--duration-fast) var(--ease-out),
              border-color var(--duration-fast) var(--ease-out),
              color var(--duration-fast) var(--ease-out),
              transform var(--duration-fast) var(--ease-out);
}
.odin-ui .btn:active:not(:disabled):not(.btn-loading) { transform: translateY(1px); }
.odin-ui .btn:disabled,
.odin-ui .btn[aria-disabled="true"] { opacity: 0.5; cursor: not-allowed; }

/* Størrelser (min-height holder ikon-only ≥24px hit-target, WCAG 2.5.8) */
.odin-ui .btn-sm { padding: var(--space-2) var(--space-3); font-size: var(--font-size-sm); min-height: 30px; }
.odin-ui .btn-md { padding: var(--space-3) var(--space-4); font-size: var(--font-size-base); min-height: 38px; }
.odin-ui .btn-lg { padding: var(--space-4) var(--space-5); font-size: var(--font-size-md); min-height: 46px; }

/* Primary — den ENE accent-flade per skærm (Solveig: one-accent-per-screen).
   Hvid på copper ~3.1:1 → kun AA som large/semibold; .btn-sm-primary undgås til normal tekst. */
.odin-ui .btn-primary { background: var(--accent); color: var(--text-on-accent); border-color: var(--accent); font-weight: var(--font-weight-semibold); }
.odin-ui .btn-primary:hover:not(:disabled):not(.btn-loading) { background: var(--accent-hover); border-color: var(--accent-hover); }

/* Secondary — workhorse, neutral outline */
.odin-ui .btn-secondary { background: var(--surface-2); color: var(--text-primary); border-color: var(--border-strong); }
.odin-ui .btn-secondary:hover:not(:disabled):not(.btn-loading) { background: var(--surface-hover); border-color: var(--border-focus); }

/* Ghost — lav emphasis / toolbar */
.odin-ui .btn-ghost { background: transparent; color: var(--text-secondary); border-color: transparent; }
.odin-ui .btn-ghost:hover:not(:disabled):not(.btn-loading) { background: var(--surface-2); color: var(--text-primary); }

/* Danger — soft-at-rest, solid på hover (intent vises ved commitment).
   Tekst = --neg #F87171 (~5.2:1 AA-PASS; #EF4444 ville være ~4.0:1 = under AA). */
.odin-ui .btn-danger { background: var(--neg-soft); color: var(--neg); border-color: var(--neg-soft); }
.odin-ui .btn-danger:hover:not(:disabled):not(.btn-loading) { background: var(--neg); color: var(--text-on-accent); border-color: var(--neg); }

/* Success — samme soft→solid-mønster */
.odin-ui .btn-success { background: var(--pos-soft); color: var(--pos); border-color: var(--pos-soft); }
.odin-ui .btn-success:hover:not(:disabled):not(.btn-loading) { background: var(--pos); color: var(--text-on-accent); border-color: var(--pos); }

/* Indre ikon-slot (makroen emitterer <span class="btn__icon">) — omdøbt fra .btn-icon
   for at frigøre .btn-icon til icon-only-varianten. */
.odin-ui .btn__icon { display: inline-flex; align-items: center; font-size: 1.05em; }
.odin-ui .btn-label { line-height: 1; }

/* MODIFIER: .btn-icon-only = icon-only (kvadratisk). KRÆVER aria-label (håndhæves i makroen).
   Navnet matcher design-manual.html (approved mål-state) og frigør .btn-icon helt.
   Stacker på en variant: class="btn btn-ghost btn-icon-only". */
.odin-ui .btn-icon-only { aspect-ratio: 1 / 1; padding: var(--space-2); min-width: 38px; }
.odin-ui .btn-icon-only.btn-sm { min-width: 30px; min-height: 30px; }
.odin-ui .btn-icon-only.btn-lg { min-width: 46px; min-height: 46px; }
.odin-ui .btn-icon-only .btn-label { display: none; }   /* sikkerheds-net hvis label medfølger */

/* MODIFIER: .btn-loading — spinner uden bredde-hop; aria-busy sættes af makro/JS.
   Skjuler label/ikon via opacity (bevarer layout), centreret ::after-spinner. */
.odin-ui .btn-loading { position: relative; pointer-events: none; cursor: wait; color: transparent !important; }
.odin-ui .btn-loading .btn__icon,
.odin-ui .btn-loading .btn-label { opacity: 0; }
.odin-ui .btn-loading::after {
  content: ""; position: absolute; top: 50%; left: 50%;
  width: 1em; height: 1em; margin: -0.5em 0 0 -0.5em;
  border: 2px solid currentColor; border-top-color: transparent; border-radius: 50%;
  color: var(--accent-text);
  animation: odin-btn-spin var(--duration-slow, 0.6s) linear infinite;
}
@keyframes odin-btn-spin { to { transform: rotate(360deg); } }
@media (prefers-reduced-motion: reduce) {
  .odin-ui .btn-loading::after { animation: none; border-top-color: currentColor; opacity: 0.6; }
}

/* ── 3. GLOBALT STICKY-NAV (additivt — bruges først når en side opter ind) ───
   Løftet fra .spaa-sticky (D-355). Border-bottom (ikke shadow — shadow laver grå
   dis over talgrid). Opaque bg så scrollede tal ikke skinner igennem. z under alle
   overlays (chat 1000 / view-as-banner 9999). top offsettes under view-as-banneret. */
.page-header-sticky,
.tab-nav-sticky {
  position: sticky;
  top: var(--sticky-top, 0px);
  z-index: var(--z-sticky, 200);
  background: var(--bg-base, #0d1014);
  border-bottom: 1px solid var(--border-strong, #232830);
}
/* View-as-banner (position:fixed top:0, 44px spacer) — skub sticky ned så fanerne
   ikke skjules bag det røde banner (a11y 2.4.11 Focus Not Obscured). */
body.view-as-active {
  --sticky-top: 44px;
}
/* Elevation-on-scroll: tilføj .is-stuck via JS når baren faktisk pinner (valgfrit). */
.tab-nav-sticky.is-stuck,
.page-header-sticky.is-stuck { box-shadow: var(--shadow-sm); }

/* 400%-zoom / lav viewport-højde: drop sticky så chrome ikke æder hele skærmen
   (WCAG 1.4.10 Reflow). */
@media (max-height: 600px) {
  .page-header-sticky,
  .tab-nav-sticky { position: static; }
}
