/* =============================================================
   SAVE STATUS — Pill (global) + Trail (pro Quest)
   Zwei Layer Feedback fuer "kommt der Tap wirklich am Server an":
   - .save-pill   floats unter der Topbar, zeigt Pending/Saved/Errors
   - .save-trail  spawned body-level am Check-Button, 3-Punkte-Phase
   Beide leben ausserhalb von #page-content, damit DOM-Re-Renders
   der Quest-Liste sie nicht killen.
   ============================================================= */

/* ── Pill ─────────────────────────────────────────────────────── */
.save-pill {
  position: fixed;
  /* Topbar-Hoehe ~64px + safe-area-inset; sitzt direkt darunter, mittig. */
  top: calc(env(safe-area-inset-top, 0px) + 64px + 6px);
  left: 50%;
  transform: translateX(-50%);
  z-index: 90;
  display: inline-flex; align-items: center; gap: 8px;
  max-width: calc(100% - 32px);
  padding: 8px 14px;
  border-radius: 999px;
  border: 1px solid var(--border);
  background: var(--surface);
  color: var(--text);
  font-family: var(--font-mono);
  font-size: var(--fs-small);
  letter-spacing: var(--tracking-wide);
  box-shadow: 0 6px 20px rgba(0, 0, 0, 0.35);
  animation: save-pill-in 200ms var(--ease) both;
  user-select: none;
}
.save-pill[hidden] { display: none; }

.save-pill-icon {
  display: inline-flex; align-items: center; justify-content: center;
  width: 18px; height: 18px;
  border-radius: 50%;
  background: var(--surface2);
  font-size: 12px; line-height: 1;
}
.save-pill-label { white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }

/* Pending: gedeckte Primary-Note, Icon pulst leicht. */
.save-pill.save-pill-pending {
  border-color: color-mix(in srgb, var(--primary) 35%, var(--border));
  color: color-mix(in srgb, var(--text) 90%, var(--primary));
}
.save-pill.save-pill-pending .save-pill-icon {
  background: color-mix(in srgb, var(--primary) 20%, var(--surface2));
  color: var(--primary);
  animation: save-pill-pulse 1s var(--ease) infinite;
}

/* Saved: gruen, faded-out durch hidden-toggle. */
.save-pill.save-pill-saved {
  border-color: color-mix(in srgb, #22c55e 40%, var(--border));
  color: #d1fae5;
}
.save-pill.save-pill-saved .save-pill-icon {
  background: color-mix(in srgb, #22c55e 20%, var(--surface2));
  color: #22c55e;
}

/* AuthError: deutlich, accent-betont, klickbar. */
.save-pill.save-pill-auth {
  border-color: color-mix(in srgb, var(--accent) 50%, var(--border));
  background: color-mix(in srgb, var(--accent) 10%, var(--surface));
  color: var(--text);
}
.save-pill.save-pill-auth .save-pill-icon {
  background: color-mix(in srgb, var(--accent) 25%, var(--surface2));
  color: var(--accent);
}

/* NetError: warnung, aber freundlicher als AuthError. */
.save-pill.save-pill-net {
  border-color: color-mix(in srgb, var(--danger) 40%, var(--border));
  color: color-mix(in srgb, var(--text) 95%, var(--danger));
}
.save-pill.save-pill-net .save-pill-icon {
  background: color-mix(in srgb, var(--danger) 20%, var(--surface2));
  color: var(--danger);
}

@keyframes save-pill-in {
  from { opacity: 0; transform: translate(-50%, -6px); }
  to   { opacity: 1; transform: translate(-50%, 0); }
}
@keyframes save-pill-pulse {
  0%, 100% { transform: scale(1);   opacity: 1;   }
  50%      { transform: scale(1.1); opacity: 0.7; }
}

/* ── Trail (3-Phasen-Indikator pro Quest-Tap) ─────────────────── */
.save-trail {
  position: fixed; z-index: 80; pointer-events: none;
  display: flex; align-items: center; gap: 6px;
  padding: 4px 8px;
  border-radius: 999px;
  background: color-mix(in srgb, var(--surface) 92%, transparent);
  border: 1px solid var(--border);
  font-family: var(--font-mono);
  font-size: var(--fs-micro);
  letter-spacing: var(--tracking-wide);
  color: var(--muted);
  box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
  animation: save-trail-in 180ms var(--ease) both;
}
.save-trail-step {
  width: 6px; height: 6px; border-radius: 50%;
  background: var(--border-strong);
  transition: background var(--dur-base) var(--ease), transform var(--dur-base) var(--ease);
}
.save-trail-step.is-active {
  background: var(--primary);
  transform: scale(1.25);
}
.save-trail-label {
  margin-left: 4px;
  white-space: nowrap;
}

/* Done: alle drei Punkte gruen, Label verblasst nach 1.2s. */
.save-trail.save-trail-done .save-trail-step.is-active {
  background: #22c55e;
}
.save-trail.save-trail-done {
  color: #86efac;
  animation: save-trail-out 1.2s var(--ease) forwards;
}

/* Error: rote Punkte, Pill bleibt 2.5s sichtbar. */
.save-trail.save-trail-error .save-trail-step {
  background: var(--danger);
}
.save-trail.save-trail-error {
  color: color-mix(in srgb, var(--text) 80%, var(--danger));
  animation: save-trail-out 2.5s var(--ease) forwards;
}

@keyframes save-trail-in {
  from { opacity: 0; transform: translateY(-4px); }
  to   { opacity: 1; transform: translateY(0); }
}
@keyframes save-trail-out {
  0%   { opacity: 1; }
  70%  { opacity: 1; }
  100% { opacity: 0; transform: translateY(-4px); }
}

/* =============================================================
   UPDATE-BANNER — fixed oben, ueberschreibt safe-area
   Erscheint wenn der Service Worker eine neue Version installed
   hat und der alte SW noch controllert (= echte Update-Situation).
   Slide-in von oben, transluzent dunkel.
   ============================================================= */
#update-banner {
  position: fixed;
  top: calc(env(safe-area-inset-top, 0px) + 8px);
  left: 50%;
  transform: translateX(-50%) translateY(-130%);
  z-index: 120;
  display: inline-flex; align-items: center; gap: 12px;
  max-width: calc(100% - 24px);
  padding: 10px 14px;
  border-radius: 999px;
  border: 1px solid color-mix(in srgb, var(--primary) 40%, var(--border));
  background: var(--surface);
  color: var(--text);
  font-family: var(--font-mono);
  font-size: var(--fs-small);
  letter-spacing: var(--tracking-wide);
  box-shadow: 0 8px 24px rgba(0, 0, 0, 0.45);
  transition: transform var(--dur-slow, 320ms) var(--ease-spring, cubic-bezier(.34,1.56,.64,1));
}
#update-banner[hidden] { display: none; }
#update-banner.open    { transform: translateX(-50%) translateY(0); }

.update-banner-text  { white-space: nowrap; }
.update-banner-btn {
  background: var(--primary);
  color: #fff;
  border: none;
  border-radius: var(--r-full);
  padding: 6px 14px;
  font: inherit; font-weight: 600;
  cursor: pointer;
  transition: opacity var(--dur-fast) var(--ease);
}
.update-banner-btn:hover  { opacity: 0.9; }
.update-banner-btn:active { opacity: 0.75; }
.update-banner-dismiss {
  background: transparent;
  color: var(--muted);
  border: none;
  font: inherit; font-size: 18px; line-height: 1;
  padding: 0 4px;
  cursor: pointer;
}
.update-banner-dismiss:hover { color: var(--text); }
@media (prefers-reduced-motion: reduce) {
  #update-banner { transition: none; }
}
