/* ───────────────────────────────────────────────────────────────────────────
   Polymarket Pulse — design system  ·  "THE ORACLE"  (PRESENTATION layer)

   Brand world: the all-seeing eye over the prediction markets. A single amber
   accent (#F2941B) glowing on a near-black ground; warm-bone ink; Space Grotesk
   for headlines and Space Mono for data + labels (prices, thresholds, eyebrows).
   Light theme is a warm "parchment" inversion that keeps the same amber.

   ⚠️ Palette + type are centralised in CSS variables — change colours only in the
   :root (light) and [data-theme="dark"] (Oracle) blocks below.
   ⚠️ After ANY edit to web/*.css|js you MUST bump ?v= in web/index.html, or the
      Telegram WebView serves a stale cached client.
   ─────────────────────────────────────────────────────────────────────────── */

@import url("https://fonts.googleapis.com/css2?family=Space+Grotesk:wght@400;500;600;700&family=Space+Mono:wght@400;700&display=swap");

:root {
  /* ── LIGHT theme — "parchment" (warm bone ground, deep-navy ink, amber accent) ── */
  --bg: #F4F1E9;            /* app background (warm bone)               */
  --card: #FFFFFF;          /* cards / plates / inputs / option rows    */
  --card-2: #FBF8F0;        /* nested / inset surfaces                  */
  --ink: #16191F;           /* primary text                            */
  --muted: #6C7280;         /* secondary text, hints (AA on --bg)       */
  --faint: #9AA0AD;         /* chevrons, disabled glyphs                */
  --accent: #D97A12;        /* amber — fills (buttons, selected)        */
  --accent-strong: #B5640A; /* amber TEXT on light (AA ≥4.5:1)          */
  --accent-ink: #16191F;    /* text/icon over an amber fill (dark)      */
  --accent-soft: rgba(217,122,18,.10); /* tinted selected backgrounds   */
  --line: #E4DFD2;          /* borders, dividers, off-switch track      */
  --line-2: #EFE9DC;        /* faint inner dividers (menu rows)         */
  --yes: #1E9E6A;           /* outcome Yes / positive move              */
  --no:  #D2452F;           /* outcome No / negative move / errors      */
  --switch-knob: #FFFFFF;   /* switch thumb                            */
  --switch-off: #9AA0AD;    /* off-state thumb                         */
  --glow: none;             /* no glow in light theme                  */
  --shadow-card: 0 1px 2px rgba(20,16,8,.05);
  --shadow-pop: 0 10px 30px -10px rgba(20,16,8,.18);
  --logo-ink: #8A8F9E;      /* (consumed via logo.svg fallback)        */
  --logo-accent: #D97A12;

  /* ── type ── */
  --font-body: -apple-system, "SF Pro Text", Roboto, "Segoe UI", system-ui, sans-serif;
  --font-head: "Space Grotesk", -apple-system, "Segoe UI", system-ui, sans-serif;
  --font-mono: "Space Mono", ui-monospace, "SF Mono", Menlo, monospace;
  /* motion + tap target (THE ORACLE reference token system) */
  --ease: cubic-bezier(.2, .6, .2, 1); --dur-fast: .12s; --dur-switch: .16s; --dur-page: .32s; --tap-min: 44px;
}

[data-theme="dark"] {
  /* ── DARK theme — "The Oracle" (near-black, amber glow) ── */
  --bg: #0B0E15;
  --card: #151922;
  --card-2: #1B202B;
  --ink: #ECE9E1;           /* warm bone */
  --muted: #868C9B;         /* AA on --bg (~5.9:1) */
  --faint: #565C6E;
  --accent: #F2941B;
  --accent-strong: #F2941B; /* amber reads cleanly on dark already */
  --accent-ink: #0B0E15;    /* near-black over amber */
  --accent-soft: rgba(242,148,27,.10);
  --line: #242A36;
  --line-2: #1A1F29;
  --yes: #34C98B;
  --no:  #F2604A;
  --switch-knob: #0B0E15;
  --switch-off: #6B7280;
  --glow: 0 0 14px rgba(242,148,27,.35);  /* amber halo on active controls */
  --shadow-card: none;
  --shadow-pop: 0 14px 40px -12px rgba(0,0,0,.7);
}

* { box-sizing: border-box; }
html, body {
  margin: 0; padding: 0;
  background: var(--bg); color: var(--ink);
  font: 16px/1.5 var(--font-body);
  -webkit-font-smoothing: antialiased;
}

/* ── topbar: back · logo · wordmark · wizard progress ── */
.topbar {
  display: flex; align-items: center; gap: 10px;
  padding: 14px 16px 10px;
  border-bottom: 1px solid var(--line-2);
}
.topback {
  width: 36px; height: 36px; border-radius: 11px; flex: none;
  border: 1px solid var(--line); background: var(--card);
  color: var(--ink); font-size: 18px; cursor: pointer; line-height: 1;
  transition: border-color .12s, background .12s;
}
.topback:hover { border-color: var(--accent); }
.topback:focus-visible { outline: 2px solid var(--accent); outline-offset: 2px; }
.topback.hidden { display: none; }
.logo { width: 28px; height: 28px; filter: drop-shadow(0 0 6px rgba(242,148,27,.0)); }
[data-theme="dark"] .logo { filter: drop-shadow(0 0 7px rgba(242,148,27,.45)); }
.brand {
  font-family: var(--font-mono); font-weight: 700;
  letter-spacing: .14em; text-transform: uppercase; font-size: 12px;
}
.progress {
  margin-inline-start: auto; font-family: var(--font-mono);
  font-size: 11px; color: var(--faint); text-align: end; letter-spacing: .04em;
}

/* RTL-languages (fa/ar): text direction follows content */
h1.title, p.hint, .info, .opt .lb, .toggle-row .lb, .ta-label, .input-label,
textarea.ta, .card-title { unicode-bidi: plaintext; }

.screen { padding: 8px 16px 104px; max-width: 560px; margin: 0 auto; }

h1.title {
  font-family: var(--font-head); font-weight: 600;
  font-size: 22px; line-height: 1.18; letter-spacing: -.01em;
  margin: 14px 0 4px;
}
p.hint { margin: 0 0 16px; color: var(--muted); font-size: 14.5px; line-height: 1.5; }

/* ── info plate / explanatory paragraph ── */
.info {
  background: var(--card); border: 1px solid var(--line);
  border-radius: 14px; padding: 14px 16px; margin: 10px 0;
  white-space: pre-wrap; font-size: 14.5px; line-height: 1.55;
  box-shadow: var(--shadow-card);
}
/* inline / fatal error plate ("⚠️ …") */
.info.err, .err {
  color: var(--no); border-color: color-mix(in oklab, var(--no) 45%, var(--line));
}
.err { font-size: 14px; margin: 8px 0; }

/* ── section divider — wide-tracked mono "eyebrow" (the deck signature) ── */
.section {
  font-family: var(--font-mono); font-weight: 700;
  font-size: 11px; letter-spacing: .2em; text-transform: uppercase;
  color: var(--muted);
  margin: 26px 2px 10px; padding-bottom: 7px;
  border-bottom: 1px solid var(--line);
}

/* ── options (choice / multichoice): icon · label · marker ── */
.options { display: flex; flex-direction: column; gap: 8px; margin: 10px 0; }
.opt {
  display: flex; align-items: center; gap: 11px;
  background: var(--card); border: 1.5px solid var(--line);
  border-radius: 14px; padding: 12px 14px; cursor: pointer;
  user-select: none; box-shadow: var(--shadow-card);
  transition: border-color .12s, background .12s, box-shadow .12s;
}
.opt:hover { border-color: var(--faint); }
.opt .ic { width: 24px; text-align: center; font-size: 16px; }
.opt .lb { flex: 1; font-size: 14.5px; }
.opt .desc { display: block; font-size: 13px; color: var(--muted); margin-top: 2px; }
.opt .mark {
  width: 22px; height: 22px; border-radius: 7px;
  border: 2px solid var(--line); display: inline-flex;
  align-items: center; justify-content: center;
  font-size: 14px; color: var(--accent-ink); flex: none;
  transition: background .12s, border-color .12s;
}
.opt.sel { border-color: var(--accent); background: var(--accent-soft); box-shadow: var(--glow); }
.opt.sel .mark { background: var(--accent); border-color: var(--accent); }
.opt.disabled { opacity: .45; cursor: default; }
.opt.disabled:hover { border-color: var(--line); }
.opt .tag {
  font-family: var(--font-mono); font-size: 10.5px; letter-spacing: .06em;
  padding: 3px 9px; border-radius: 999px;
  background: var(--card-2); border: 1px solid var(--line); color: var(--muted);
}

/* ── toggle-row: label + switch (instant-save) ── */
.toggle-row {
  display: flex; align-items: center; gap: 12px;
  background: var(--card); border: 1px solid var(--line);
  border-radius: 14px; padding: 14px 16px; margin: 8px 0; cursor: pointer;
  box-shadow: var(--shadow-card);
  transition: border-color .12s;
}
.toggle-row:hover { border-color: var(--faint); }
.toggle-row .lb { flex: 1; font-size: 15px; }
.toggle-row .desc { display: block; font-size: 13px; color: var(--muted); margin-top: 2px; }
/* ⚙️ opens the module's detail screen (does NOT flip the switch) */
.toggle-row .row-cfg { flex: none; margin-right: 2px; opacity: .55; font-size: 15px; cursor: pointer; }
.toggle-row .row-cfg:hover { opacity: 1; }
.switch {
  width: 48px; height: 28px; border-radius: 999px; flex: none;
  background: var(--line); position: relative; transition: background .16s;
}
.switch::after {
  content: ""; position: absolute; top: 3px; inset-inline-start: 3px;   /* RTL-aware */
  width: 22px; height: 22px; border-radius: 50%;
  background: var(--switch-off);
  transition: inset-inline-start var(--dur-switch, .16s) var(--ease), background var(--dur-switch, .16s) var(--ease);
  box-shadow: 0 1px 3px rgba(0,0,0,.3);
}
.toggle-row.on .switch { background: var(--accent); box-shadow: var(--glow); }
.toggle-row.on .switch::after { inset-inline-start: 23px; background: var(--switch-knob); }
/* disabled toggle (roadmap feature) — greyed, inert */
.toggle-row.disabled { opacity: .5; cursor: default; pointer-events: none; }
/* nesting (under a section header in settings / wizard) */
.toggle-row.nested, .segment.nested, .ta-label.nested {
  margin-inline-start: 12px; border-inline-start: 3px solid var(--line);
  padding-inline-start: 14px; border-radius: 0 14px 14px 0;
}
.toggle-row.nested { padding-top: 11px; padding-bottom: 11px; margin: 6px 0 6px; margin-inline-start: 12px; }
.toggle-row.nested .lb { font-size: 14.5px; }

/* ── segment: row of pill toggles (thresholds, frequency, scope) ── */
.segment { display: flex; gap: 7px; margin: 4px 0 10px; flex-wrap: wrap; }
.segbtn {
  flex: 1 1 0; min-width: max-content;
  padding: 11px 8px; border-radius: 11px;
  font-family: var(--font-mono); font-size: 13px;
  border: 1.5px solid var(--line); background: var(--card);
  color: var(--ink); cursor: pointer; text-align: center;
  transition: border-color .12s, background .12s, color .12s;
}
.segbtn:hover { border-color: var(--accent); }
.segbtn:focus-visible { outline: 2px solid var(--accent); outline-offset: 2px; }
.segbtn.on {
  border-color: var(--accent); background: var(--accent);
  color: var(--accent-ink); font-weight: 700; box-shadow: var(--glow);
}

/* ── text area (text_area) ── */
.ta-label { margin: 14px 0 6px; font-weight: 600; font-size: 14.5px; }
/* small grey caption above a single-line field (text_input label) */
.input-label {
  margin: 16px 0 6px;
  font-family: var(--font-mono); letter-spacing: .04em;
  color: var(--muted); font-size: 12.5px;
}
textarea.ta {
  width: 100%; min-height: 76px; resize: vertical;
  background: var(--card); color: var(--ink);
  border: 1.5px solid var(--line); border-radius: 14px;
  padding: 12px 14px; font: inherit; font-size: 15px; line-height: 1.5;
  transition: border-color .12s, box-shadow .12s;
}
textarea.ta:focus {
  outline: none; border-color: var(--accent);
  box-shadow: 0 0 0 4px var(--accent-soft);
}

/* ── single-line input (text_input): wallet / search / trader handle ── */
.input {
  width: 100%;
  padding: 13px 15px;
  border: 1.5px solid var(--line);
  border-radius: 14px;
  font-family: var(--font-mono);
  font-size: 16px;             /* ≥16px → iOS won't zoom on focus */
  letter-spacing: .01em;
  background: var(--card);
  color: var(--ink);
  margin: 8px 0;
  transition: border-color .12s, box-shadow .12s;
}
.input::placeholder { color: var(--faint); }
.input:focus {
  outline: none; border-color: var(--accent);
  box-shadow: 0 0 0 4px var(--accent-soft);
}

/* ── buttons ── */
.btn {
  width: 100%; border: 1.5px solid transparent; border-radius: 12px; padding: 14px;
  font-family: var(--font-head); font-size: 16px; font-weight: 600;
  cursor: pointer; margin-top: 12px;
  transition: filter .12s, transform .06s, border-color .12s, background .12s, color .12s;
}
.btn:active { transform: translateY(1px); }
.btn:focus-visible { outline: 2px solid var(--accent); outline-offset: 2px; }
.btn.primary {
  background: var(--accent); color: var(--accent-ink);
  box-shadow: 0 8px 26px -8px color-mix(in oklab, var(--accent) 70%, transparent);
}
.btn.primary:hover { filter: brightness(1.07); }
.btn.ghost {
  background: transparent; color: var(--muted); border: 1.5px solid var(--line);
}
.btn.ghost:hover { border-color: var(--accent); color: var(--ink); }
.btn.block { display: block; width: 100%; margin: 14px 0 6px; }
.btn:disabled { opacity: .5; cursor: default; transform: none; filter: none; }

/* ── bottom fallback navigation (outside Telegram: web mode) ── */
.fallback-nav {
  position: fixed; bottom: 0; left: 0; right: 0;
  display: flex; gap: 10px; padding: 12px 16px calc(12px + env(safe-area-inset-bottom));
  background: color-mix(in oklab, var(--bg) 90%, transparent);
  backdrop-filter: blur(10px);
  border-top: 1px solid var(--line);
}
.fallback-nav.hidden { display: none; }
.fallback-nav .btn { flex: 1; margin-top: 0; }

/* ── signal card (market / trade feed; tap → Polymarket) ── */
.card {
  display: block; position: relative; text-decoration: none;
  color: var(--ink); background: var(--card);
  border: 1px solid var(--line);
  border-radius: 14px; padding: 13px 14px 13px 16px; margin: 8px 0;
  box-shadow: var(--shadow-card);
  transition: border-color .12s, transform .08s, box-shadow .12s;
}
/* thin amber spine — the "watched" feed identity */
.card::before {
  content: ""; position: absolute; left: 0; top: 10px; bottom: 10px;
  width: 3px; border-radius: 0 3px 3px 0;
  background: var(--accent); opacity: .55;
}
.card:hover { border-color: var(--accent); transform: translateY(-1px); }
.card:hover::before { opacity: 1; }
.card-title { font-family: var(--font-head); font-weight: 600; font-size: 14.5px; line-height: 1.3; padding-right: 22px; }
.card-meta { margin-top: 6px; font-family: var(--font-mono); font-size: 12px; color: var(--muted); }
/* news-context line (mover) — italic, muted */
.card-news { margin-top: 5px; font-style: italic; color: var(--muted); font-size: 13px; }
.card-foot { margin-top: 5px; color: var(--muted); font-size: 12.5px; }
.card-img { width: 28px; height: 28px; border-radius: 50%; vertical-align: middle; margin-right: 8px; object-fit: cover; }
/* corner ✖ (watchlist / traders remove) */
.card-x {
  position: absolute; top: 8px; right: 10px; line-height: 1;
  color: var(--faint); font-size: 14px; padding: 4px 6px; cursor: pointer; border-radius: 6px;
  transition: color .12s, background .12s;
}
.card-x:hover { color: var(--no); background: color-mix(in oklab, var(--no) 12%, transparent); }

/* ── global search (widget search): input + results ── */
.searchbox { margin: 6px 0 10px; }
.search-inp {
  width: 100%; background: var(--card); color: var(--ink);
  border: 1.5px solid var(--line); border-radius: 14px;
  padding: 13px 15px; font-family: var(--font-mono); font-size: 16px;
  transition: border-color .12s, box-shadow .12s;
}
.search-inp::placeholder { color: var(--faint); }
.search-inp:focus { outline: none; border-color: var(--accent); box-shadow: 0 0 0 4px var(--accent-soft); }
.search-res { max-height: 46vh; overflow-y: auto; margin-top: 8px; }
.search-res .opt { margin-bottom: 6px; }
.search-row .desc { display: block; font-size: 12px; color: var(--muted); }

/* ── misc ── */
.center { text-align: center; }
.done-logo { width: 84px; height: 84px; margin: 26px auto 8px; display: block; }
[data-theme="dark"] .done-logo { filter: drop-shadow(0 0 22px rgba(242,148,27,.5)); }

/* build version footer (webview cache diagnostics) */
.buildtag {
  margin: 30px auto 10px; text-align: center;
  font-family: var(--font-mono); font-size: 10px; line-height: 1.7;
  letter-spacing: .1em; color: var(--faint); opacity: .5;
}

/* ── wizard steps — header + nested-content left rule ── */
.screen.wizard h1.title {
  font-family: var(--font-mono); font-weight: 700;
  font-size: 11px; letter-spacing: .2em; text-transform: uppercase; color: var(--muted);
  margin: 20px 2px 10px; padding-bottom: 7px; border-bottom: 1px solid var(--line);
}
.screen.wizard p.hint, .screen.wizard .options, .screen.wizard .bulkbar,
.screen.wizard .toggle-row, .screen.wizard .segment, .screen.wizard .ta-label,
.screen.wizard textarea.ta, .screen.wizard .info {
  margin-inline-start: 12px; border-inline-start: 3px solid var(--line);
  padding-inline-start: 14px;
}
.screen.wizard .toggle-row { padding-top: 11px; padding-bottom: 11px; }

/* sub-headers inside an option list (e.g. market categories) */
.optsec {
  font-family: var(--font-mono); font-weight: 700;
  font-size: 11px; letter-spacing: .14em; text-transform: uppercase; color: var(--faint);
  margin: 14px 2px 2px; padding-bottom: 4px;
  border-bottom: 1px solid var(--line);
}
.optsec:first-child { margin-top: 2px; }

/* bulk select (select all / none) */
.bulkbar { display: flex; gap: 8px; margin: 6px 0 8px; }
.bulkbar .mini {
  padding: 8px 13px; border-radius: 10px;
  font-family: var(--font-mono); font-size: 12px; letter-spacing: .04em;
  border: 1.5px solid var(--line); background: var(--card);
  color: var(--ink); cursor: pointer; transition: border-color .12s;
}
.bulkbar .mini:hover { border-color: var(--accent); }

/* ═══════════════════════════════════════════════════════════════════════════
   CONVICTA-specific components, Oracle-styled (the design system targets a
   slightly different widget set; these cover Convicta's renderer extras):
   👁 group drill-down · inline word/dedup/RSS icon buttons · modals · words &
   quiet config · loading spinner · SVG chrome icons · focus-visible · RTL.
   ═══════════════════════════════════════════════════════════════════════════ */

/* SVG chrome icons (currentColor, theme-aware; pointer-events off so the click
   falls through to the parent button — renderer reads ev.target) */
.icon { width: 1.15em; height: 1.15em; display: inline-block; vertical-align: -0.15em;
  fill: none; stroke: currentColor; stroke-width: 2; stroke-linecap: round;
  stroke-linejoin: round; flex: none; pointer-events: none; }
.cap-n { font-family: var(--font-mono); font-size: 12px; font-weight: 700;
  margin-inline-start: 2px; pointer-events: none; }
.dot { color: var(--accent); font-weight: 700; margin-inline-start: 1px; pointer-events: none; }

/* inline icon buttons on option rows (👁 / 📝 / 🧮 / ⚡) — ≥40px tap target */
.opt .peek, .member .stbtn, .opt .gstbtn, .opt .catstbtn, .opt .capbtn, .opt .rssmode {
  min-width: var(--tap-min, 44px); height: var(--tap-min, 44px); padding: 0 8px;
  border-radius: 10px; flex: none;
  display: inline-flex; align-items: center; justify-content: center;
  border: 1.5px solid var(--line); background: var(--card-2); color: var(--ink);
  font-family: var(--font-mono); font-size: 13px; font-weight: 700; cursor: pointer;
  transition: border-color var(--dur-fast, .12s) var(--ease), color var(--dur-fast, .12s) var(--ease); }

/* edit_menu navigation rows: faint › chevron (not a checkbox box) — matches the
   THE ORACLE reference (menu rows navigate, they don't select) */
.opt .chev { flex: none; margin-inline-start: 4px; color: var(--faint);
  font-family: var(--font-head); font-size: 22px; line-height: 1; }
[dir="rtl"] .opt .chev { transform: scaleX(-1); }
.opt .peek:hover, .opt .gstbtn:hover, .opt .catstbtn:hover, .opt .capbtn:hover,
.opt .rssmode:hover, .member .stbtn:hover { border-color: var(--accent); }

/* 👁 group drill-down panel + member rows */
.members { margin: 6px 0 6px; margin-inline-start: 14px; padding: 8px 10px;
  border-inline-start: 2px solid var(--line); border-radius: 0 12px 12px 0;
  background: var(--card-2); }
.members.hidden { display: none; }
.members-title { font-family: var(--font-mono); font-size: 11px; letter-spacing: .1em;
  text-transform: uppercase; color: var(--muted); margin: 2px 0 8px; }
.member { display: flex; align-items: center; gap: 10px; padding: 10px 6px; min-height: 44px;
  cursor: pointer; border-radius: 9px; opacity: .5; transition: opacity .12s; }
.member.on { opacity: 1; }
.member .mark { width: 20px; height: 20px; border-radius: 6px; flex: none;
  border: 2px solid var(--line); display: inline-flex; align-items: center;
  justify-content: center; font-size: 11px; color: var(--accent-ink); }
.member.on .mark { background: var(--accent); border-color: var(--accent); }
.member .lb { flex: 1; min-width: 0; font-size: 14px; }
.member .desc { display: block; font-size: 12.5px; color: var(--muted); }

/* per-source/group/category words forms (cat_tags + nested stform) */
.stform { margin: 2px 0 8px; margin-inline-start: 30px; }
.stform.hidden { display: none; }
.stform textarea.ta { min-height: 38px; margin-bottom: 4px; }
.btn.wordsbtn { display: inline-block; width: auto; padding: 11px 14px; min-height: 44px;
  margin: 6px 0 4px; font-size: 13px; font-family: var(--font-mono); letter-spacing: .04em; }
.btn.wordsbtn.nested { margin-inline-start: 12px; }
.ta-label.small { margin: 8px 0 4px; font-size: 12.5px; font-family: var(--font-mono);
  letter-spacing: .04em; color: var(--muted); }
textarea.ta.small { min-height: 40px; padding: 9px 12px; font-size: 14px; }
div.nested { margin-inline-start: 12px; border-inline-start: 3px solid var(--line);
  padding-inline-start: 14px; }

/* 🚫 unsubscribe pill inside the RSS frequency row */
.segbtn.unsub { flex: 0 0 auto; padding: 11px 13px; opacity: .85; }
.segbtn.unsub:hover { border-color: var(--no); color: var(--no); }

/* Story Cap / Quiet-hours modal popups */
.modal-ov { position: fixed; inset: 0; z-index: 50;
  background: color-mix(in oklab, var(--bg) 30%, rgba(0,0,0,.55));
  display: flex; align-items: center; justify-content: center; padding: 24px; }
.modal-card { width: 100%; max-width: 340px; background: var(--card); color: var(--ink);
  border: 1px solid var(--line); border-radius: 16px; padding: 18px 16px;
  box-shadow: var(--shadow-pop); }
.modal-title { font-family: var(--font-head); font-size: 16px; font-weight: 600; margin: 0 0 8px; }
.modal-card input.capinput { width: 100%; background: var(--card-2); color: var(--ink);
  border: 1.5px solid var(--line); border-radius: 12px; padding: 12px 14px;
  font-family: var(--font-mono); font-size: 18px; text-align: center; }
.modal-card input.capinput:focus-visible { outline: none; border-color: var(--accent);
  box-shadow: 0 0 0 4px var(--accent-soft); }
.modal-actions { display: flex; gap: 8px; margin-top: 14px; }
.modal-actions .btn { margin-top: 0; }

/* loading spinner between screens */
.spinner-ov { position: fixed; inset: 0; z-index: 40;
  background: color-mix(in oklab, var(--bg) 55%, transparent);
  display: flex; align-items: center; justify-content: center; }
.spinner { width: 34px; height: 34px; border-radius: 50%;
  border: 3px solid var(--line); border-top-color: var(--accent);
  animation: spin .7s linear infinite; }
@keyframes spin { to { transform: rotate(360deg); } }

/* 89/ soft page transition — fade on every screen swap (.32s). Re-triggered from
   app.js render() via reflow. Auto-disabled by reduced-motion below.
   95/ 21.06: OPACITY-ONLY — убран transform/translateY: на #screen он создавал
   «мёртвую зону» для тапов в Telegram WebView (кнопки не нажимались). Прозрачность
   тачи не блокирует. */
@keyframes screen-in {
  from { opacity: 0; }
  to   { opacity: 1; } }
.screen.screen-anim { animation: screen-in var(--dur-page, .32s) var(--ease) both; }

/* a11y: visible focus ring on every interactive control + reduced motion */
.opt:focus-visible, .toggle-row:focus-visible, .member:focus-visible,
.peek:focus-visible, .stbtn:focus-visible, .gstbtn:focus-visible,
.catstbtn:focus-visible, .capbtn:focus-visible, .rssmode:focus-visible,
.search-inp:focus-visible, textarea.ta:focus-visible,
[tabindex]:focus-visible, [role="button"]:focus-visible {
  outline: 2.5px solid var(--accent); outline-offset: 2px; }
@media (prefers-reduced-motion: reduce) {
  *, *::before, *::after { animation-duration: .001ms !important;
    animation-iteration-count: 1 !important; transition-duration: .001ms !important;
    scroll-behavior: auto !important; } }
