/* ── Reset & Variables ─────────────────────────────────────────────────────── */
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }

:root {
  --bg:         #0d1117;
  --bg-card:    #161b22;
  --bg-panel:   #1c2128;
  --border:     #30363d;
  --text:       #c9d1d9;
  --text-dim:   #8b949e;
  --accent:     #58a6ff;
  --accent2:    #3fb950;
  --danger:     #f85149;
  --warning:    #d29922;
  --success:    #3fb950;
  --caution:    #ff7b72;
  --sim:        #8957e5;
  --radius:     8px;
  --font:       'Segoe UI', system-ui, sans-serif;

  /* Полоса прокрутки — общая палитра для всех скроллбаров страницы */
  --sb-track:        transparent;
  --sb-thumb:        #30363d;       /* = --border */
  --sb-thumb-hover:  #484f58;       /* = чуть светлее */
}

/* ── Глобальная стилизация полосы прокрутки в стиле страницы ───────────────── */
* {
  scrollbar-width: thin;
  scrollbar-color: var(--sb-thumb) var(--sb-track);
}
*::-webkit-scrollbar         { width: 8px; height: 8px; }
*::-webkit-scrollbar-track   { background: var(--sb-track); }
*::-webkit-scrollbar-thumb   {
  background: var(--sb-thumb);
  border-radius: 4px;
  border: 2px solid var(--bg);   /* «отступ» от краев — выглядит тоньше */
}
*::-webkit-scrollbar-thumb:hover { background: var(--sb-thumb-hover); }
*::-webkit-scrollbar-corner  { background: var(--sb-track); }

/* Справка использует тот же корневой font-size, что и рабочие экраны —
   без масштабирования (раньше тут было 18px, отчего справка выглядела
   «увеличенной»). Лёгкий акцент на читабельность даёт .card { font-size:
   1.02rem } ниже, без общего зума страницы. */

/* ═══════════════════════════════════════════════════════════════════════════
   Единый стиль страниц справки (page-docs).
   Применяется ко всем разделам: /help, /training, /system, /maneuvers,
   /hardware, /about. Цель — одинаковая типографика, цветовая иерархия,
   читаемые таблицы, акцентные подзаголовки h3 голубым.
   ═══════════════════════════════════════════════════════════════════════════ */


/* Базовый размер текста и цвет — обычный текст серый, акценты ярче. */
body.page-docs .card,
body.page-docs .help-card {
  font-size: 1.02rem;          /* +1-2 пт от дефолта */
}
body.page-docs .card p,
body.page-docs .card li,
body.page-docs .card td,
body.page-docs .card .hint,
body.page-docs .help-card p,
body.page-docs .help-card__desc,
body.page-docs .subtitle {
  color: var(--text-dim);
  line-height: 1.5;
}

/* Акценты (strong, th, заголовки) — яркие. */
body.page-docs .card strong,
body.page-docs .card th,
body.page-docs .card h1,
body.page-docs .card h2,
body.page-docs .card h4,
body.page-docs .help-card__title,
body.page-docs .help-card h2 {
  color: var(--text);
}

/* Размеры заголовков. */
body.page-docs .card h2 { font-size: 1.25rem; margin-top: 1rem; }
body.page-docs .card h4 { font-size: 1rem;    margin-top: 0.7rem; }

/* h3 — подзаголовок: акцентный голубой, чуть крупнее обычного текста. */
body.page-docs .card h3,
body.page-docs .alg-card h3 {
  color: var(--accent);
  font-size: 1.08rem;
  margin-top: 0.9rem;
}

/* Код — яркий с фоном, того же размера что окружение. */
body.page-docs code {
  background: var(--bg-panel);
  padding: 0.05rem 0.3rem;
  border-radius: 3px;
  color: var(--text);
  font-size: 1em;
}

/* Таблицы справки — выравнивание ячеек + светлая шапка. */
body.page-docs .card table {
  width: 100%;
  border-collapse: collapse;
  margin: 0.5rem 0;
  font-size: 0.98rem;
}
body.page-docs .card th,
body.page-docs .card td {
  border: 1px solid var(--border);
  padding: 0.45rem 0.65rem;
  text-align: left;
  vertical-align: top;
}
body.page-docs .card th { background: var(--bg-panel); }

/* Списки внутри карточек — нормальный отступ под маркеры. */
body.page-docs .card ol,
body.page-docs .card ul {
  padding-left: 1.6rem;
  margin: 0.4rem 0;
}
body.page-docs .card ol ul,
body.page-docs .card ul ul,
body.page-docs .card ol ol { margin: 0.3rem 0; }
body.page-docs .card li { margin-bottom: 0.25rem; }

/* Блоки `<pre>` — подсветка как в /hardware и /maneuvers. */
body.page-docs .card pre {
  background: #0d1117;
  border: 1px solid var(--border);
  border-radius: 6px;
  padding: 0.7rem 0.9rem;
  overflow-x: auto;
  font-family: monospace;
  font-size: 0.95rem;
  line-height: 1.45;
  white-space: pre;
  color: #c9d1d9;
  margin: 0.5rem 0;
}
body.page-docs .card pre .c { color: #7ee787; }   /* комментарии — зелёный */
body.page-docs .card pre .k { color: #ff7b72; }   /* keywords    — красный */
body.page-docs .card pre .s { color: #a5d6ff; }   /* строки      — голубой */

/* Callout «note» — общий стиль для всех справок. */
body.page-docs .card .note {
  background: color-mix(in srgb, var(--accent) 8%, transparent);
  border-left: 3px solid var(--accent);
  padding: 0.6rem 0.85rem;
  border-radius: 4px;
  margin: 0.6rem 0;
}
body.page-docs .card .warn {
  background: color-mix(in srgb, #f0883e 10%, transparent);
  border-left: 3px solid #f0883e;
  padding: 0.6rem 0.85rem;
  border-radius: 4px;
  margin: 0.6rem 0;
}

/* TOC (оглавление) — одинаковый стиль в /maneuvers и /hardware. */
body.page-docs .card .toc {
  font-size: 0.95rem;
  line-height: 1.6;
  color: var(--text-dim);
  margin-bottom: 1rem;
}
body.page-docs .card .toc a { color: var(--accent); text-decoration: none; }
body.page-docs .card .toc a:hover { text-decoration: underline; }

body {
  font-family: var(--font);
  background:  var(--bg);
  color:       var(--text);
  min-height:  100vh;
  display:     flex;
  flex-direction: column;
}

/* ── Topbar ─────────────────────────────────────────────────────────────────── */
/* ВАЖНО: padding и border-bottom оставляем компактными, иначе ломается
   calc(100vh - 6rem) у .control-layout — появляется лишняя прокрутка.
   Заметность держим фоном-градиентом и тенью — они не меняют высоту. */
.topbar {
  display:         flex;
  align-items:     center;
  gap:             1.5rem;
  padding:         0.6rem 1.4rem;
  background:      linear-gradient(180deg, #1c2330 0%, #161b22 100%);
  border-bottom:   1px solid var(--border);
  box-shadow:      0 3px 10px rgba(0, 0, 0, 0.55);
  position:        sticky;
  top: 0;
  z-index: 100;
}

.topbar__brand { display: flex; align-items: baseline; gap: 0.3rem; }
.brand-vega { font-size: 1.5rem; font-weight: 800; color: var(--accent); letter-spacing: -1px; }
.brand-rex  { font-size: 1.5rem; font-weight: 800; color: var(--accent2); letter-spacing: -1px; }
.brand-sub  { font-size: 0.7rem; color: var(--text-dim); margin-left: 0.5rem; }

/* ── Хаб-страница «Справка»: 3 карточки разделов ────────────────────── */
.help-grid {
  display: grid; gap: 1.4rem;
  grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
  margin-top: 1.6rem;
}
/* Каждая карточка получает свой акцент через CSS-переменную --card-accent
   (задаётся inline-style в HTML), это влияет на полоску, hover-бордер и
   фоновый ореол иконки. */
.help-card {
  position: relative; overflow: hidden;
  background: var(--surface);
  border: 1px solid var(--border);
  border-radius: 10px;
  padding: 1.7rem 1.5rem 1.4rem;
  display: flex; flex-direction: column;
  transition: transform 0.18s ease, border-color 0.18s, box-shadow 0.18s;
}
.help-card::before {
  content: ''; position: absolute; left: 0; right: 0; top: 0;
  height: 3px;
  background: var(--card-accent, var(--accent));
  opacity: 0.85;
}
.help-card:hover {
  transform: translateY(-3px);
  border-color: var(--card-accent, var(--accent));
  box-shadow: 0 12px 28px rgba(0,0,0,0.40);
}
.help-card__icon {
  display: inline-flex; align-items: center; justify-content: center;
  width: 56px; height: 56px;
  font-size: 2rem; line-height: 1;
  background: color-mix(in srgb, var(--card-accent, var(--accent)) 15%, transparent);
  border-radius: 12px;
  margin-bottom: 1rem;
}
.help-card__title {
  font-size: 1.25rem; font-weight: 700;
  margin: 0 0 0.7rem;
  color: var(--text);
  letter-spacing: -0.2px;
}
.help-card__desc {
  flex: 1;
  margin: 0 0 1.3rem;
  font-size: 0.88rem;
  color: var(--text-dim);
  line-height: 1.55;
}
.help-card__desc code {
  background: var(--bg-panel); padding: 1px 5px; border-radius: 3px;
  font-size: 0.8rem; color: var(--accent2);
}
.help-card .btn {
  align-self: flex-start;
  padding: 0.55rem 1.1rem;
  background: var(--card-accent, var(--accent));
  color: #fff;
}
.help-card .btn:hover { opacity: 0.9; }

/* ── Страница «О программе» ──────────────────────────────────────────
   Композиция: hero сверху, ниже последовательность секций-карточек
   с акцентной полосой слева (--sec-accent). Сетки tech/features/audience
   адаптивные (auto-fit). Стиль перекликается с help-card, но это уже
   контент-страница, не хаб, поэтому без grid 3-в-ряд по умолчанию. */
.about-page {
  max-width: 1100px;
  margin: 0 auto;
  padding: 0.4rem 0.6rem 2rem;
  display: flex; flex-direction: column;
  gap: 1.6rem;
}

/* Hero */
.about-hero {
  position: relative; overflow: hidden;
  background:
    radial-gradient(circle at 18% 20%, rgba(88,166,255,0.18) 0%, transparent 55%),
    radial-gradient(circle at 82% 30%, rgba(163,113,247,0.18) 0%, transparent 55%),
    linear-gradient(180deg, #1c2330 0%, #161b22 100%);
  border: 1px solid var(--border);
  border-radius: 14px;
  padding: 2rem 2rem 1.8rem;
  text-align: center;
}
.about-hero__badge {
  display: inline-block;
  padding: 0.25rem 0.8rem;
  font-size: 0.72rem; letter-spacing: 0.06em; text-transform: uppercase;
  color: var(--accent);
  background: color-mix(in srgb, var(--accent) 13%, transparent);
  border: 1px solid color-mix(in srgb, var(--accent) 35%, transparent);
  border-radius: 999px;
  margin-bottom: 1rem;
}
.about-hero__title {
  font-size: clamp(2.5rem, 6vw, 4rem);
  font-weight: 800;
  letter-spacing: -2px;
  margin: 0.2rem 0 0.8rem;
  line-height: 1;
}
.about-hero__title .brand-vega,
.about-hero__title .brand-rex {
  font-size: inherit;
}
.about-hero__tagline {
  max-width: 720px;
  margin: 0 auto 1.2rem;
  font-size: 1.02rem;
  color: var(--text-dim);
  line-height: 1.55;
}
.about-hero__tagline strong { color: var(--text); }
.about-hero__meta {
  display: flex; flex-wrap: wrap; gap: 0.5rem; justify-content: center;
  margin-top: 0.4rem;
}
.about-chip {
  font-size: 0.74rem;
  padding: 0.28rem 0.7rem;
  border-radius: 999px;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  color: var(--text-dim);
}
.about-chip--accent {
  background: color-mix(in srgb, var(--accent2) 18%, transparent);
  border-color: color-mix(in srgb, var(--accent2) 45%, transparent);
  color: var(--accent2);
  font-weight: 600;
}

/* Секции */
.about-section {
  position: relative;
  background: var(--bg-card);
  border: 1px solid var(--border);
  border-radius: 12px;
  padding: 1.5rem 1.6rem 1.4rem;
  transition: border-color 0.18s, box-shadow 0.18s;
}
.about-section::before {
  content: '';
  position: absolute; left: 0; top: 18px; bottom: 18px;
  width: 3px;
  background: var(--sec-accent, var(--accent));
  border-radius: 2px;
}
.about-section:hover {
  border-color: color-mix(in srgb, var(--sec-accent, var(--accent)) 60%, var(--border));
  box-shadow: 0 8px 24px rgba(0,0,0,0.30);
}
.about-section__head {
  display: flex; align-items: center; gap: 0.8rem;
  margin-bottom: 1rem;
}
.about-section__icon {
  display: inline-flex; align-items: center; justify-content: center;
  width: 42px; height: 42px;
  font-size: 1.4rem; line-height: 1;
  background: color-mix(in srgb, var(--sec-accent, var(--accent)) 16%, transparent);
  border: 1px solid color-mix(in srgb, var(--sec-accent, var(--accent)) 35%, transparent);
  border-radius: 10px;
}
.about-section__title {
  font-size: 1.25rem;
  font-weight: 700;
  letter-spacing: -0.3px;
  margin: 0;
}
.about-section__body {
  color: var(--text-dim);
  font-size: 0.92rem;
  line-height: 1.3;
}
.about-section__body p + p { margin-top: 0.7rem; }
.about-section__body strong { color: var(--text); font-weight: 600; }

.about-goal {
  font-size: 1.02rem !important;
  color: var(--text) !important;
  padding: 0.8rem 1rem;
  background: color-mix(in srgb, var(--sec-accent, var(--accent)) 8%, transparent);
  border-left: 3px solid var(--sec-accent, var(--accent));
  border-radius: 0 8px 8px 0;
  margin-bottom: 1rem;
}
.about-list {
  margin: 0; padding-left: 1.4rem;
  display: flex; flex-direction: column; gap: 0.4rem;
}
.about-list li::marker { color: var(--sec-accent, var(--accent)); }

/* Возможности — сетка плиток.
   На широких экранах 3 колонки; если последняя карточка осталась одна в
   ряду (количество карточек ≡ 1 mod 3 — например 10) — центрируем её.
   На средних — 2 колонки, тогда 10 карточек ложатся ровно 5×2.
   На узких — одна колонка. */
.about-features {
  display: grid;
  grid-template-columns: repeat(3, 1fr);
  gap: 1rem;
}
.about-features > .about-feature:last-child:nth-child(3n+1) {
  grid-column-start: 2;
}
@media (max-width: 1000px) {
  .about-features { grid-template-columns: repeat(2, 1fr); }
  .about-features > .about-feature:last-child:nth-child(3n+1) {
    grid-column-start: auto;
  }
}
@media (max-width: 600px) {
  .about-features { grid-template-columns: 1fr; }
}
.about-feature {
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 10px;
  padding: 1.1rem 1.1rem 1rem;
  transition: transform 0.18s ease, border-color 0.18s;
}
.about-feature:hover {
  transform: translateY(-2px);
  border-color: color-mix(in srgb, var(--sec-accent, var(--accent)) 60%, var(--border));
}
.about-feature__icon {
  font-size: 1.6rem;
  margin-bottom: 0.5rem;
}
.about-feature h3 {
  font-size: 0.98rem;
  font-weight: 700;
  margin: 0 0 0.4rem;
  color: var(--text);
}
.about-feature p {
  font-size: 0.83rem;
  line-height: 1.5;
  color: var(--text-dim);
  margin: 0;
}

/* Технологии */
.about-tech-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
  gap: 0.6rem;
}
.about-tech {
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-left: 3px solid var(--sec-accent, var(--accent));
  border-radius: 6px;
  padding: 0.6rem 0.85rem;
  transition: background 0.15s;
}
.about-tech:hover {
  background: color-mix(in srgb, var(--sec-accent, var(--accent)) 8%, var(--bg-panel));
}
.about-tech__name {
  font-size: 0.92rem;
  font-weight: 700;
  color: var(--text);
  margin-bottom: 0.1rem;
}
.about-tech__role {
  font-size: 0.78rem;
  color: var(--text-dim);
  line-height: 1.4;
}

/* Аудитория */
.about-audience {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(240px, 1fr));
  gap: 1rem;
}
.about-audience__card {
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 10px;
  padding: 1.2rem 1.1rem 1rem;
  text-align: center;
  transition: transform 0.18s, border-color 0.18s;
}
.about-audience__card:hover {
  transform: translateY(-2px);
  border-color: color-mix(in srgb, var(--sec-accent, var(--accent)) 55%, var(--border));
}
.about-audience__icon {
  font-size: 2rem;
  margin-bottom: 0.5rem;
}
.about-audience__card h3 {
  font-size: 0.95rem;
  font-weight: 700;
  margin: 0 0 0.4rem;
  color: var(--text);
}
.about-audience__card p {
  font-size: 0.82rem;
  color: var(--text-dim);
  line-height: 1.5;
  margin: 0;
}

/* Автор проекта — компактная строка с маленьким кружком «1Т» */
.about-author-section {
  background:
    linear-gradient(135deg,
      color-mix(in srgb, var(--sec-accent, var(--accent)) 10%, var(--bg-card)) 0%,
      var(--bg-card) 60%);
  padding: 0.7rem 1.2rem;
}
.about-author {
  display: flex; align-items: center; gap: 0.85rem;
}
.about-author__avatar {
  flex-shrink: 0;
  width: 38px; height: 38px;
  display: flex; align-items: center; justify-content: center;
  border-radius: 50%;
  font-weight: 800;
  font-size: 0.8rem;
  letter-spacing: 0.5px;
  color: #fff;
  background:
    linear-gradient(135deg, var(--sec-accent, var(--accent)) 0%, #6b46c1 100%);
  box-shadow: 0 2px 8px rgba(0,0,0,0.3);
}
.about-author-line {
  margin: 0;
  font-size: 0.85rem;
  font-weight: 400;
  color: var(--text-dim);
}
.about-author__text {
  display: flex; flex-direction: column; gap: 0.3rem;
}
.about-author-note {
  margin: 0;
  font-size: 0.85rem;
  font-weight: 400;
  line-height: 1.4;
  color: var(--text-dim);
}

/* Подвал страницы (отдельно от глобального .appbar) */
.about-footer {
  display: flex; justify-content: center; align-items: center;
  gap: 0.6rem;
  margin-top: 0.6rem;
  padding: 0.8rem;
  font-size: 0.85rem;
  color: var(--text-dim);
}
.about-footer__name {
  font-weight: 700;
  letter-spacing: 0.5px;
  background: linear-gradient(90deg, var(--accent) 0%, var(--accent2) 100%);
  -webkit-background-clip: text;
  background-clip: text;
  color: transparent;
}
.about-footer__dot { opacity: 0.4; }

/* Адаптив под узкие экраны: автор в колонку */
@media (max-width: 520px) {
  .about-author { flex-direction: column; text-align: center; gap: 0.7rem; }
}

/* ── Статус-подвал ──────────────────────────────────────────────────────
   Полностью непрозрачный, фиксирован внизу. Содержит статус-строку слева
   и версию справа. Контент основного грида (control-layout) отнимает
   высоту подвала из своего расчёта, поэтому пересечения нет. */
.appbar {
  position: fixed; left: 0; right: 0; bottom: 0;
  height: 22px; padding: 0 0.7rem;
  background: #161b22;                        /* solid, не прозрачно */
  border-top: 1px solid var(--border);
  box-shadow: 0 -2px 6px rgba(0,0,0,0.35);    /* лёгкая тень над контентом */
  display: flex; align-items: center; gap: 0.6rem;
  font-size: 0.7rem; line-height: 1; color: var(--text-dim);
  z-index: 60;
}
.appbar__status {
  flex: 1; min-width: 0;
  white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
  /* Базовый цвет — голубой (как кнопка «Инспектор»). Уровни
     info/success/warning не перебивают цвет режима (как просил
     пользователь): только режим робота определяет тон, а ошибки
     и подсказки выделяются явно. */
  color: var(--accent);
}
/* В режиме «осторожно» — жёлтый, как кнопка Осторожно. */
.appbar--cautious .appbar__status { color: var(--warning); }
/* Sticky-подсказка (Режим зон) — серая, нейтральная подсказка. */
.appbar__status--hint    { color: var(--text-dim) !important; }
/* Ошибки — красные, перебивают режим. */
.appbar__status--error   { color: var(--danger)  !important; }
.appbar__version {
  flex-shrink: 0;
  padding-left: 0.7rem; border-left: 1px solid var(--border);
  color: var(--text-dim); opacity: 0.75;
  font-size: 0.75rem; font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

/* Резерв снизу — высота appbar; контент не уезжает под подвал. */
body { padding-bottom: 22px; }

.topbar__nav { display: flex; gap: 0.25rem; flex: 1; }
.nav-link {
  padding: 0.35rem 0.8rem;
  border-radius: var(--radius);
  color: var(--text-dim);
  text-decoration: none;
  font-size: 0.85rem;
  font-weight: 500;
  transition: background 0.15s, color 0.15s, border-color 0.15s;
  border: 1px solid transparent;
}
.nav-link:hover      { background: rgba(255,255,255,0.04); color: var(--text); }
.nav-link--active {
  /* Тёмный фон с акцент-бордером — лучше контраст, чем сплошной голубой. */
  background: rgba(88,166,255,0.10);
  color: var(--accent) !important;
  border-color: rgba(88,166,255,0.35);
  font-weight: 600;
}

.topbar__status { display: flex; align-items: center; gap: 0.5rem; }

/* ── Badges ─────────────────────────────────────────────────────────────────── */
.badge {
  font-size: 0.7rem; font-weight: 600;
  padding: 0.15rem 0.5rem; border-radius: 99px;
}
.badge--online  { background: #1a4731; color: var(--success); }
.badge--offline { background: #3d1a1a; color: var(--danger); }
.badge--sim     { background: #2c1f52; color: var(--sim); }
.badge--caution { background: #4d2f00; color: var(--warning); }
/* Чип режима подключения: SIM (симулятор) / LOC (локальный мост) / NET (туннель) */
.badge--mode-sim { background: #2c1f52; color: var(--sim); }
.badge--mode-loc { background: #15303f; color: #58a6ff; }
.badge--mode-net { background: #3d2a00; color: #f0883e; }

/* ── Main content ────────────────────────────────────────────────────────────── */
.main-content { flex: 1; padding: 1rem; overflow: hidden; }
/* Подвал убран из base.html — стиль оставлен на случай возврата. */
.footer { display: none; }

/* ── Control layout ─────────────────────────────────────────────────────────── */
.control-layout {
  display: grid;
  grid-template-columns: minmax(240px, 280px) minmax(0, 1fr) minmax(288px, 384px);
  gap: 0.75rem;
  /* Topbar ≈ 3.5rem, main-content padding 1rem×2 = 2rem, appbar 22px.
     Итого вычитаем из 100vh, чтобы основные панели + статус-подвал
     поместились без вертикального скролла. */
  height: calc(100vh - 5.5rem - 22px);
  min-width: 0;
}

.side-panel, .log-panel, .viz-container {
  display: flex;
  flex-direction: column;
  gap: 0.75rem;
  overflow-x: hidden;
  min-width: 0;
}
.viz-container { overflow-y: auto; }
/* Правая панель: код-карточка фиксированной высоты сверху, журнал ниже
   занимает всё оставшееся место и скроллится внутри своей .cmd-log
   (как в левой панели для side-panel__scroll). Внешнего скролла нет. */
.log-panel       { overflow: hidden; }
.log-panel > .card             { flex-shrink: 0; }      /* карточка кода фиксирована */
.log-panel > .card--full       { flex: 1; min-height: 0; }  /* журнал тянется */

/* Левая панель: верх (state + СТОП) фиксирован, низ скроллится отдельно */
.side-panel { overflow: hidden; }
.side-panel__sticky {
  flex-shrink: 0;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.side-panel__scroll {
  flex: 1;
  display: flex;
  flex-direction: column;
  gap: 0.75rem;
  overflow-y: auto;
  overflow-x: hidden;
  /* Полосу прокрутки всегда показываем тонкой — иначе ширина панели
     прыгает при наведении/уходе. Стиль наследуется от глобального * правила. */
  scrollbar-width: thin;
  scrollbar-gutter: stable;
}

/* Скроллбар внутри журнала — скрытый по умолчанию, появляется на hover. */
.cmd-log                     { scrollbar-width: none; }
.cmd-log::-webkit-scrollbar  { width: 0; height: 0; }
.cmd-log:hover               { scrollbar-width: thin; }
.cmd-log:hover::-webkit-scrollbar { width: 8px; }

/* Скроллбары редактора кода — скрыты, пока не наведёшь курсор: так они
   не перекрывают код при чтении. Вынести их ЗА рамку окна нельзя —
   браузерный скроллбар принадлежит самому прокручиваемому элементу. */
.program-code                     { scrollbar-width: none; }
.program-code::-webkit-scrollbar  { width: 0; height: 0; }
.program-code:hover               { scrollbar-width: thin; }
.program-code:hover::-webkit-scrollbar { width: 9px; height: 9px; }

.side-panel { max-width: 280px; }
.log-panel { max-width: 384px; }

/* Кнопки в quick-extra должны корректно ужиматься в узкой колонке */
.quick-extra .btn { min-width: 0; flex-shrink: 1; }

/* Большая аварийная кнопка СТОП в закрепленной шапке */
.btn--stop-big {
  width: 100%;
  padding: 0.55rem;
  font-size: 0.95rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  background: #3d1a1a;
  color: var(--danger);
  border: 1px solid #5a2222;
}
.btn--stop-big:hover { background: #4a1f1f; opacity: 1; }

/* ■ СТОП / ⏸ Пауза / ▶ Продолжить — три кнопки в одном ряду.
   Equal flex, disabled-вариант полупрозрачный. min-width:0 + overflow
   ellipsis позволяют узкой левой панели вместить все три. */
.control-row {
  display: flex;
  gap: 0.25rem;
}
.control-btn {
  flex: 1 1 0;
  min-width: 0;
  padding: 0.55rem 0.2rem;
  font-size: 1.1rem;
  font-weight: 700;
  letter-spacing: 0;
  line-height: 1;
}
.control-btn:disabled {
  opacity: 0.4;
  cursor: not-allowed;
}

/* Карточка с состоянием — компактнее, без нижнего margin */
.card--state { padding: 0.6rem 0.75rem; }

/* Карточка с группами управления */
.card--ctrl { padding: 0.5rem 0.75rem 0.6rem; }

/* ── Раскрывающиеся группы кнопок управления ──────────────────────────────── */
.ctrl-group {
  border-top: 1px solid var(--border);
  padding: 0.25rem 0 0.35rem;
}
.ctrl-group:first-of-type { border-top: none; }
.ctrl-group > summary {
  cursor: pointer;
  list-style: none;
  font-size: 0.65rem;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: 0.05em;
  color: var(--text-dim);
  padding: 0.25rem 0;
  user-select: none;
  display: flex;
  align-items: center;
  gap: 0.4rem;
  transition: color 0.15s;
}
.ctrl-group > summary:hover { color: var(--text); }
.ctrl-group > summary::-webkit-details-marker { display: none; }
.ctrl-group > summary::before {
  content: '▸';
  display: inline-block;
  font-size: 0.6rem;
  color: var(--text-dim);
  transition: transform 0.15s;
}
.ctrl-group[open] > summary::before { transform: rotate(90deg); }
.ctrl-group > .quick-extra,
.ctrl-group > .compass-grid { margin-top: 0.25rem; }

/* ── Cards ──────────────────────────────────────────────────────────────────── */
.card {
  background: var(--bg-card);
  border: 1px solid var(--border);
  border-radius: var(--radius);
  padding: 0.75rem;
}
.card--full { flex: 1; display: flex; flex-direction: column; }
.card__title {
  font-size: 0.75rem;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  color: var(--text-dim);
  margin-bottom: 0.6rem;
  padding-bottom: 0.4rem;
  border-bottom: 1px solid var(--border);
}

/* ── State grid ─────────────────────────────────────────────────────────────── */
.state-grid {
  display: grid;
  grid-template-columns: 1fr 1fr;
  gap: 0.4rem;
}
.state-item {
  display: flex;
  align-items: baseline;
  gap: 0.2rem;
  padding: 0.22rem 0.4rem;
  background: var(--bg-panel);
  border-radius: 5px;
}
/* Размеры увеличены на ~2пт (≈0.17rem) — крупнее значения и подписи. */
.state-label { font-size: 0.79rem; color: var(--text-dim); min-width: 38px; }
.state-value { font-size: 0.99rem; font-weight: 700; color: var(--accent); line-height: 1.1; }
.state-unit  { font-size: 0.77rem; color: var(--text-dim); }

/* ── Buttons ─────────────────────────────────────────────────────────────────── */
.btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 0.3rem;
  padding: 0.45rem 0.9rem;
  border: 1px solid transparent;     /* единая толщина обводки у всех */
  border-radius: 6px;
  font-size: 0.85rem;
  font-weight: 400;                  /* кнопки — обычный шрифт, без жирного */
  cursor: pointer;
  transition: background 0.15s, border-color 0.15s, color 0.15s, transform 0.1s;
  background: var(--border);
  color: var(--text);
  text-decoration: none;             /* для <a class="btn"> */
  font-family: inherit;
}
/* Унифицированная система кнопок:
   - Базовая толщина обводки: 1px у всех (transparent → виден цвет варианта)
   - Шрифт обычный (400), без жирного
   - Hover: бордер чуть ярче, фон чуть ярче (без opacity-трюков)
   - Active (нажатие): легкий «давим» через scale */
.btn:active     { transform: scale(0.97); }

.btn--primary   { background: var(--accent);  color: #fff; border-color: var(--accent); }
.btn--primary:hover { background: #4090ee; border-color: #4090ee; }

.btn--voice     { background: #21262d; border-color: var(--accent); color: var(--accent); width: 100%; }
.btn--voice.listening { background: #1a2f4a; border-color: var(--accent2); color: var(--accent2); }

.btn--stop      { background: #3d1a1a; color: var(--danger); border-color: rgba(248,81,73,0.35); }
.btn--stop:hover{ background: #4a1f1f; border-color: var(--danger); }

.btn--center    { background: #1a2430; color: var(--accent); border-color: rgba(88,166,255,0.25); }
.btn--center:hover { background: #1f2a38; border-color: var(--accent); }

/* Тонированные «состояния»: приглушены в покое, ярче по hover, активны = .is-active */
.btn--info      { background: #14202f; color: #4a6e96; border-color: #1e2c40; }
.btn--info:hover    { background: #1a2a3d; color: var(--accent); border-color: rgba(88,166,255,0.4); }
.btn--info.is-active{ background: #1f4068; color: #fff; border-color: var(--accent); }

.btn--warn      { background: #2a1d05; color: #7a6231; border-color: #3a2a0f; }
.btn--warn:hover    { background: #3a280a; color: var(--warning); border-color: rgba(245,184,40,0.4); }
.btn--warn.is-active{ background: #6b4200; color: #fff; border-color: var(--warning); }

.btn--danger    { background: #3d1a1a; color: var(--caution); border-color: rgba(248,81,73,0.3); }
.btn--danger:hover    { background: #4a1f1f; color: var(--danger); border-color: rgba(248,81,73,0.55); }
.btn--danger.is-active{ background: #6b2424; color: #fff; border-color: var(--danger); }

/* Кнопка «Автопилот» — фиолетовая. */
.btn--autopilot         { background: #2a1f44; color: #b794f6; border-color: rgba(137,87,229,0.30); }
.btn--autopilot:hover   { background: #392a5e; color: #cdb4ff; border-color: rgba(137,87,229,0.55); }
.btn--autopilot.is-active { background: #5a3fa0; color: #fff; border-color: #8957e5; }
/* Кнопка «Обстановка»: в покое — очень тёмная, активна — умеренно
   светлее. Контраст за счёт тёмного покоя, без кричащей яркости. */
/* «Обстановка» и «Поле» — один оттенок (ориентир — «Обстановка»). */
#btn-zone-mode,
#btn-reset-field          { background: #241010; }
#btn-zone-mode:hover,
#btn-reset-field:hover    { background: #3a1818; }

/* «Поле» и «Зарядить» — одинаковый размер (min-width на обе кнопки). */
#btn-charge,
#btn-reset-field          { min-width: 6.5rem; }
#btn-zone-mode.is-active  {
  background: #7a2a2a;
  color: #fff;
  border-color: var(--danger);
  box-shadow: 0 0 7px rgba(248,81,73,0.40);
}

.btn--success   { background: #16321f; color: var(--success); border-color: #1f4d2c; }
.btn--success:hover { background: #1d4028; color: #5fcc7e; border-color: rgba(63,185,80,0.5); }
.btn--sm      { padding: 0.3rem 0.6rem; font-size: 0.78rem; }
.btn--xs      { padding: 0.2rem 0.45rem; font-size: 0.72rem; }

/* Подсветка значения в карточке состояния */
.state-value--cautious { color: var(--warning) !important; }
.state-value--low      { color: var(--warning) !important; }
.state-value--crit     { color: var(--danger) !important; }

/* ── Модалка-предупреждение от сервера (блок команды и т.п.) ───────────── */
.warn-modal {
  position: fixed;
  inset: 0;
  z-index: 1500;
  background: rgba(13, 17, 23, 0.78);
  display: flex;
  align-items: center;
  justify-content: center;
  padding: 1rem;
}
.warn-modal[hidden] { display: none; }
.warn-modal__panel {
  background: var(--bg-card);
  border: 1px solid var(--border);
  border-left: 4px solid var(--warning);
  border-radius: var(--radius);
  max-width: 520px;
  width: 100%;
  box-shadow: 0 20px 60px rgba(0, 0, 0, 0.6);
}
.warn-modal__header {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  padding: 0.85rem 1.1rem 0.5rem;
}
.warn-modal__icon  { font-size: 1.4rem; color: var(--warning); }
.warn-modal__title { font-size: 1rem; font-weight: 600; }
.warn-modal__body  { padding: 0.5rem 1.1rem 1rem; color: var(--text-dim); line-height: 1.45; }
.warn-modal__footer {
  display: flex;
  justify-content: flex-end;
  padding: 0 1.1rem 1.1rem;
}

/* «Робот думает»: бейдж под карточкой состояния. Фиолетовая иконка во время
   планирования, красная если решение не найдено. */
.thinking-badge {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  margin-top: 0.5rem;
  padding: 0.4rem 0.6rem;
  background: #2c1f52;
  color: #b48cff;
  border: 1px solid #4a3585;
  border-radius: var(--radius);
  font-size: 0.78rem;
  font-weight: 600;
}
.thinking-badge__icon {
  font-size: 1.1rem;
  animation: thinking-pulse 1.2s ease-in-out infinite;
}
.thinking-badge--failed {
  background: #3d1a1a;
  color: var(--danger);
  border-color: #5a2222;
}
.thinking-badge--failed .thinking-badge__icon {
  animation: none;
}
@keyframes thinking-pulse {
  0%, 100% { opacity: 1.0; transform: scale(1); }
  50%      { opacity: 0.55; transform: scale(0.9); }
}

/* Шкала заряда батареи — как у реальных индикаторов: зеленый → желтый → красный */
.state-item--battery {
  grid-column: span 2;             /* во всю ширину карточки состояния */
  align-items: center;
  gap: 0.5rem;
}
.battery-gauge {
  flex: 1;
  position: relative;
  height: 16px;
  background: var(--bg);
  border: 1px solid var(--border);
  border-radius: 4px;
  overflow: hidden;
  min-width: 80px;
}
.battery-gauge__fill {
  position: absolute;
  inset: 0 auto 0 0;
  width: 100%;
  /* Те же темные оттенки, что у кнопок — .btn--success / --warn / --danger */
  background: #16321f;            /* как фон кнопки 🔋 Зарядить */
  transition: width 0.3s ease, background-color 0.3s ease;
}
.battery-gauge__fill--mid  { background: #3d3500; }   /* темно-оливковый */
.battery-gauge__fill--low  { background: #4d2f00; }   /* как .btn--warn */
.battery-gauge__fill--crit { background: #3d1a1a; }   /* как .btn--danger */
.battery-gauge__pct {
  position: absolute;
  inset: 0;
  display: flex;
  align-items: center;
  justify-content: center;
  font-size: 0.72rem;
  font-weight: 700;
  /* Цвет текста = цвет соответствующей зоны: яркий поверх темного фона */
  color: var(--success);
  text-shadow: 0 0 2px rgba(0,0,0,0.7);
  z-index: 1;
  transition: color 0.3s ease;
}
.state-item--battery .battery-gauge__fill--mid  ~ .battery-gauge__pct,
.state-item--battery:has(.battery-gauge__fill--mid) .battery-gauge__pct  { color: #d4c324; }
.state-item--battery:has(.battery-gauge__fill--low) .battery-gauge__pct  { color: var(--warning); }
.state-item--battery:has(.battery-gauge__fill--crit) .battery-gauge__pct { color: var(--danger); }

/* ── Quick controls ─────────────────────────────────────────────────────────── */
.quick-grid {
  display: grid;
  grid-template-columns: repeat(3, 1fr);
  gap: 0.35rem;
  margin-bottom: 0.5rem;
}
.quick-grid .btn--quick {
  aspect-ratio: 1;
  font-size: 1.1rem;
  background: var(--bg-panel);
  border: 1px solid var(--border);
}
.quick-grid--sm .btn--quick {
  aspect-ratio: unset;
  font-size: 0.75rem;
  padding: 0.2rem 0;
  height: 28px;
}
.quick-grid--sm { gap: 0.2rem; margin-bottom: 0.35rem; }

.quick-row-label {
  font-size: 0.65rem;
  color: var(--text-dim);
  text-transform: uppercase;
  letter-spacing: 0.05em;
  margin: 0.4rem 0 0.15rem;
}

.quick-extra {
  display: flex;
  flex-wrap: wrap;
  gap: 0.3rem;
}
.quick-extra.quick-extra--right {
  justify-content: flex-end;
}

/* ── Voice ─────────────────────────────────────────────────────────────────── */
.hint   { font-size: 0.78rem; color: var(--text-dim); margin-bottom: 0.5rem; }
.voice-status {
  margin-top: 0.5rem;
  font-size: 0.78rem;
  line-height: 1.3;
  color: var(--text-dim);
  /* Фиксированная высота ровно под 2 строки — не зависит от длины
     текста, панель «Голосовое управление» не дёргается. Длинные
     сообщения обрезаются многоточием.
     box-sizing: border-box (глобальный reset) → padding входит в height,
     поэтому к 2.6em (2 строки) прибавляем 0.6rem паддинга, иначе вторая
     строка обрезается. */
  height: calc(2.6em + 0.6rem);
  min-height: calc(2.6em + 0.6rem);
  max-height: calc(2.6em + 0.6rem);
  overflow: hidden;
  display: -webkit-box;
  -webkit-box-orient: vertical;
  -webkit-line-clamp: 2;
  padding: 0.3rem 0;
  border-top: 1px solid var(--border);
}

/* ── Command form ────────────────────────────────────────────────────────────── */
.cmd-form { display: flex; gap: 0.4rem; }
.cmd-input {
  flex: 1;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 6px;
  padding: 0.4rem 0.6rem;
  color: var(--text);
  font-size: 0.85rem;
}
.cmd-input:focus { outline: none; border-color: var(--accent); }

/* ── Viz container ───────────────────────────────────────────────────────────── */
.viz-container {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  background: var(--bg-card);
  border: 1px solid var(--border);
  border-radius: var(--radius);
  overflow: hidden;
}
.viz-toolbar {
  display: flex;
  flex-direction: column;
  row-gap: 0.4rem;
  padding: 0.5rem 0.75rem;
  border-bottom: 1px solid var(--border);
  flex-shrink: 0;
}
/* Строка тулбара: флажки — на первой, кнопки интерфейса — на второй. */
.viz-toolbar__row {
  display: flex;
  align-items: center;
  flex-wrap: wrap;
  row-gap: 0.4rem;
  column-gap: 0.5rem;
}
.viz-toolbar__sep {
  width: 1px;
  align-self: stretch;
  background: var(--border);
  margin: 0 0.25rem;
}
.toggle { display: flex; align-items: center; gap: 0.3rem; font-size: 0.78rem; cursor: pointer; }
.toggle input { accent-color: var(--accent); }
/* Флажки в тулбаре (Сетка/Путь/Дальномер, Опасные зоны/Зоны внимания) —
   тем же шрифтом, что и подпись «Препятствия:» (наследуют размер строки). */
.viz-toolbar__row .toggle { font-size: inherit; }
/* Цвет подписей галочек «Препятствия»: опасные — красным, внимания — жёлтым. */
.obst-label--danger    { color: var(--danger); }
.obst-label--attention { color: var(--warning); }
#robot-canvas { flex: 1; width: 100%; display: block; min-width: 0; }

/* ── Log panel ───────────────────────────────────────────────────────────────── */
.cmd-log {
  flex: 1;
  overflow-y: auto;
  scrollbar-width: thin;
  font-size: 0.85rem;
  display: flex;
  flex-direction: column;
  gap: 0.1rem;              /* вдвое меньше: было 0.2rem */
}
.log-entry {
  padding: 0.1rem 0.4rem;   /* вдвое меньше по вертикали: было 0.25rem */
  border-radius: 4px;
  border-left: 2px solid var(--border);
  line-height: 1.25;        /* плотнее: было 1.3 */
  /* Многострочные сообщения (traceback Python) — сохраняем переносы строк
     как в IDE-консоли, но позволяем длинным строкам перенестись по ширине. */
  white-space: pre-wrap;
  word-break: break-word;
  font-family: inherit;
}
.log-entry--info    { border-color: var(--accent); }
.log-entry--success { border-color: var(--success); color: var(--success); }
.log-entry--warning { border-color: var(--warning); color: var(--warning); }
.log-entry--error   {
  border-color: var(--danger);
  color:        var(--danger);
  font-family:  "Consolas", "Menlo", monospace;  /* traceback моноширинно */
}
.log-time { color: var(--text-dim); margin-right: 0.3rem; }

/* Карточка задания: вставляется в журнал по клику «📋 Задание».
   Содержит либо описание активной миссии (+ кнопку завершить досрочно),
   либо подсказку «миссия не загружена» с инструкцией где её найти. */
.log-task-card {
  padding: 0.7rem 0.9rem;
  margin: 0.5rem 0;
  background: rgba(88,166,255,0.07);
  border: 1px solid rgba(88,166,255,0.35);
  border-left-width: 3px;
  border-radius: 6px;
  font-size: 0.82rem;
  line-height: 1.5;
  color: var(--text);
}
.log-task-card__title {
  font-weight: 600; margin-bottom: 0.35rem;
}
.log-task-card__body {
  margin: 0; padding: 0;
  color: var(--text-dim);
  font-family: inherit;
  font-size: 0.82rem;
  white-space: pre-wrap;
  background: transparent;
  border: 0;
}
.log-task-card__body a       { color: var(--accent); text-decoration: none; }
.log-task-card__body a:hover { text-decoration: underline; }
.log-task-card__body strong  { color: var(--text); }
.log-task-card__actions {
  margin-top: 0.5rem;
  display: flex; gap: 0.4rem;
  flex-wrap: wrap;
}
/* Карточка после mission_finished — кнопки заблокированы, бейдж итога. */
.log-task-card--finalized { opacity: 0.85; }
.log-task-card--finalized button[disabled] {
  cursor: not-allowed; opacity: 0.45;
}
.task-verdict {
  margin-left: 0.5rem;
  padding: 1px 7px;
  border-radius: 10px;
  font-size: 0.72rem;
  font-weight: 600;
  vertical-align: middle;
}
.task-verdict--ok   { background: #16321f; color: var(--success); border: 1px solid #1f4d2c; }
.task-verdict--fail { background: #3d1a1a; color: var(--danger);  border: 1px solid rgba(248,81,73,0.35); }

/* ── Страница статистики ────────────────────────────────────────────── */
.stats-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
  gap: 0.6rem;
  margin-top: 0.5rem;
}
.stats-item {
  display: flex; flex-direction: column;
  padding: 0.7rem 0.9rem;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 6px;
}
.stats-label {
  font-size: 0.78rem; color: var(--text-dim); margin-bottom: 0.25rem;
}
.stats-value {
  font-size: 1.4rem; font-weight: 600; color: var(--text);
  font-variant-numeric: tabular-nums;
}
.stats-value--accent { color: var(--warning); }

.stats-table {
  width: 100%;
  border-collapse: collapse;
  margin-top: 0.5rem;
  font-size: 0.9rem;
  table-layout: auto;
}
.stats-table th,
.stats-table td {
  padding: 0.7rem 1.1rem;       /* больше воздуха между колонками */
  text-align: left;
  border-bottom: 1px solid var(--border);
  white-space: nowrap;          /* шапка не переносится «Лучшее\nкачество» */
}
.stats-table th {
  color: var(--text-dim);
  font-weight: 500;
  font-size: 0.78rem;
  text-transform: uppercase;
  letter-spacing: 0.03em;
}
.stats-table tbody tr:hover { background: rgba(63,185,80,0.05); }
/* Узкая колонка № — фиксированная по контенту, не растягивается. */
.stats-table__col-num { width: 1%; text-align: right; }
.stats-table__num     { text-align: right; color: var(--text-dim);
                        font-family: ui-monospace, Consolas, monospace; }
/* Колонка «Миссия» забирает остаток ширины. */
.stats-table th:nth-child(2),
.stats-table td:nth-child(2) { white-space: normal; width: auto; }
.stats-table__stars strong { color: var(--warning); }
/* Ссылки в статистике — зелёные (включая :visited, иначе браузер
   подсвечивает посещённые фиолетовым по умолчанию). */
.stats-table a,
.stats-table a:visited { color: var(--success); text-decoration: none; }
.stats-table a:hover   { color: #5fcc7e; text-decoration: underline; }

/* ── Card title row (title + button) ────────────────────────────────────────── */
.card__title-row {
  display: flex;
  align-items: center;
  justify-content: space-between;
  margin-bottom: 0.6rem;
  padding-bottom: 0.4rem;
  border-bottom: 1px solid var(--border);
}
.card__title-row .card__title {
  margin-bottom: 0;
  padding-bottom: 0;
  border-bottom: none;
}

.program-actions {
  display: flex;
  gap: 0.3rem;
  align-items: center;
}

/* Кнопка «</>» — открывает модалку «Сборка кода для робота» */
.btn--code {
  font-family: ui-monospace, "SF Mono", Consolas, monospace;
  font-size: 0.85rem;
  letter-spacing: -0.02em;
}

/* Модалка «Сборка кода для робота» (отдельная от fullscreen-editor) */
.code-export-modal {
  position: fixed; inset: 0;
  z-index: 1000;
  display: flex; align-items: center; justify-content: center;
}
.code-export-modal[hidden] { display: none; }
.code-export-modal__backdrop {
  position: absolute; inset: 0;
  background: rgba(0, 0, 0, 0.55);
  backdrop-filter: blur(2px);
}
.code-export-modal__panel {
  position: relative;
  width: min(900px, calc(100vw - 2rem));
  max-height: calc(100vh - 3rem);
  background: var(--bg-card);
  border: 1px solid var(--border);
  border-radius: 10px;
  box-shadow: 0 18px 50px rgba(0, 0, 0, 0.45);
  display: flex; flex-direction: column;
}
.code-export-modal__header {
  display: flex; align-items: center; justify-content: space-between;
  gap: 1rem;
  padding: 0.85rem 1rem 0.6rem;
  border-bottom: 1px solid var(--border);
}
.code-export-modal__title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 600;
  color: var(--text);
}
.code-export-modal__actions {
  display: flex; gap: 0.35rem; align-items: center;
}
.code-export-modal__hint {
  margin: 0;
  padding: 0.55rem 1rem 0;
  font-size: 0.78rem;
  line-height: 1.45;
  color: var(--text-dim);
}
/* Редактор с гуттером строк — sticky-нумерация слева от textarea. */
.code-export-modal__editor {
  display: flex;
  margin: 0.6rem 1rem 1rem;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 6px;
  max-height: calc(100vh - 12rem);
  overflow: hidden;
  font-family: ui-monospace, "SF Mono", Consolas, monospace;
  font-size: 0.82rem;
  line-height: 1.45;
  tab-size: 4;
}
.code-export-modal__gutter {
  margin: 0;
  padding: 0.75rem 0.5rem 0.75rem 0.7rem;
  background: color-mix(in srgb, var(--bg-panel) 60%, var(--bg-card));
  border-right: 1px solid var(--border);
  color: var(--text-dim);
  text-align: right;
  user-select: none;
  white-space: pre;
  overflow: hidden;
  font-family: inherit;
  font-size: inherit;
  line-height: inherit;
  min-width: 2.5rem;
  flex-shrink: 0;
}
.code-export-modal__textarea {
  flex: 1;
  margin: 0;
  padding: 0.75rem 0.9rem;
  background: transparent;
  border: none;
  outline: none;
  resize: none;
  color: var(--text);
  font-family: inherit;
  font-size: inherit;
  line-height: inherit;
  tab-size: inherit;
  white-space: pre;
  overflow: auto;
}
.code-export-modal__textarea:focus {
  outline: none;
  box-shadow: inset 0 0 0 2px color-mix(in srgb, var(--accent) 40%, transparent);
}

/* Шестерёнка скорости визуализации в шапке панели кода */
.sim-speed-wrap { position: relative; display: inline-flex; }
.sim-speed-wrap .btn {
  display: inline-flex; align-items: baseline; gap: 0.18rem;
  padding-left: 0.45rem; padding-right: 0.45rem;
}
.sim-speed__icon { font-size: 0.95rem; line-height: 1; }
.sim-speed__label {
  font-size: 0.78rem;
  color: var(--text-dim);
  min-width: 1.6rem;
  text-align: left;
}
.sim-speed-wrap--active .sim-speed__label { color: var(--accent); }
.sim-speed-menu {
  position: absolute;
  top: calc(100% + 4px);
  right: 0;
  z-index: 30;
  display: flex; flex-direction: column;
  min-width: 4.2rem;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 6px;
  box-shadow: 0 6px 18px rgba(0,0,0,0.35);
  padding: 0.25rem;
  gap: 0.15rem;
}
.sim-speed-menu[hidden] { display: none; }
.sim-speed-menu__item {
  background: transparent;
  border: 1px solid transparent;
  color: var(--text);
  font: inherit;
  font-size: 0.85rem;
  padding: 0.22rem 0.5rem;
  border-radius: 4px;
  cursor: pointer;
  text-align: center;
}
.sim-speed-menu__item:hover { background: var(--bg-card); border-color: var(--border); }
.sim-speed-menu__item--active {
  background: color-mix(in srgb, var(--accent) 18%, transparent);
  border-color: var(--accent);
  color: var(--accent);
}
/* Описание поля Python-кода (над textarea) — мелкое, чтобы не отвлекало */
.code-desc {
  font-size: 0.7rem;
  color: var(--text-dim);
  margin-bottom: 0.4rem;
  line-height: 1.3;
}

.program-code {
  width: 100%;
  font-family: 'Courier New', monospace;
  font-size: 0.78rem;          /* боковая панель */
  line-height: 1.35;
  border: 1px solid var(--border);
  border-radius: 4px;
  /* Доп. отступ снизу — чтобы последняя строка кода не липла к рамке. */
  padding: 0.5rem 0.5rem 0.85rem;
  color: var(--text-dim);
  background: var(--bg-panel);
  min-height: 10rem;
  resize: vertical;
  /* Отключаем перенос строк — code editor convention. textarea имеет
     wrap="off" в HTML, overlay тоже использует white-space: pre. Так
     длинные строки скроллятся горизонтально, и каждый символ имеет
     ОДИНАКОВУЮ координату в textarea и overlay → клик попадает точно
     в нужный символ, выделение работает предсказуемо.
     scrollbar-gutter: stable резервирует место под вертикальный
     скроллбар, чтобы вертикальная позиция строк совпадала. */
  white-space: pre;
  overflow: auto;
  scrollbar-gutter: stable;
}
.program-code:not(:empty) { color: var(--text); }

/* ── Overlay-подсветка синтаксиса для program-code ──────────────────────── */
/* Поверх textarea рисуется <pre> с подсвеченным HTML; сам textarea имеет
   прозрачный текст, но видимый каретку и выделение. Поведение редактора
   полностью сохранено, добавлено только цветовое выделение комментариев. */
.code-wrap {
  position: relative;
  width: 100%;
}
.code-wrap--modal {
  flex: 1;
  display: flex;
  min-height: 0;
}

.code-overlay {
  position: absolute;
  inset: 0;
  margin: 0;
  font-family: 'Courier New', monospace;
  font-size: 0.78rem;          /* должно совпадать с .program-code */
  line-height: 1.35;
  padding: 0.5rem 0.5rem 0.85rem;   /* совпадает с .program-code */
  border: 1px solid transparent;
  border-radius: 4px;
  color: var(--text);
  background: var(--bg-panel);
  /* white-space: pre + НЕТ wrap. textarea (wrap="off") ведёт себя так же,
     поэтому каждая строка занимает ОДИНАКОВУЮ позицию в обоих слоях →
     клик в textarea точно попадает на символ, который видит пользователь
     в overlay. Горизонтальный скролл — на textarea, overlay сдвигается
     через CSS transform (см. syncScroll в control.js). */
  white-space: pre;
  /* Overlay не скроллится сам — внутренний <code> сдвигается через
     CSS transform по scrollTop textarea (см. JS sync). Так не получается
     второй скроллбар, и нет глюка с программным scrollTop в Webkit. */
  overflow: hidden;
  pointer-events: none;
  z-index: 1;
}
.code-overlay code {
  display: block;
  will-change: transform;
}
.code-overlay code { font: inherit; color: inherit; background: none; padding: 0; }
.program-code--editor {
  position: relative;
  z-index: 2;
  background: transparent !important;
  color: transparent !important;
  caret-color: var(--text) !important;
}
.program-code--editor::selection {
  background: rgba(88, 166, 255, 0.35);
  color: transparent;
}
/* Зеленые комментарии (как в стандартных редакторах кода) */
.hl-comment { color: #6a9955; font-style: italic; }

/* ── Модальное окно «Python код во весь экран» ─────────────────────────── */
.code-modal {
  position: fixed;
  inset: 0;
  z-index: 1000;
  background: rgba(13, 17, 23, 0.85);
  display: flex;
  align-items: stretch;
  justify-content: center;
  padding: 2rem;
}
.code-modal[hidden] { display: none; }
.code-modal__panel {
  background: var(--bg-card);
  border: 1px solid var(--border);
  border-radius: var(--radius);
  flex: 1;
  display: flex;
  flex-direction: column;
  max-width: 1200px;
  box-shadow: 0 20px 60px rgba(0, 0, 0, 0.6);
}
/* Toggle-класс «развёрнуто на весь экран»: убираем padding (2rem) и
   max-width (1200px) у внутренней панели — она растягивается на весь
   viewport, строка кода получает максимально возможную ширину.
   Браузерный хром (URL-бар, табы) при этом ОСТАЁТСЯ — мы не зовём
   Fullscreen API, просто перекрашиваем разметку. */
.code-modal--maximized {
  padding: 0;
}
.code-modal--maximized .code-modal__panel {
  max-width: none;
  border: none;
  border-radius: 0;
  box-shadow: none;
}
.code-modal__header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  padding: 0.75rem 1rem;
  border-bottom: 1px solid var(--border);
}
.code-modal__title {
  font-size: 0.85rem;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  color: var(--text-dim);
}
.program-code--modal {
  flex: 1;
  /* font-size управляется CSS-переменной --code-font-size, чтобы Shift+колесо
     могло её менять. Значение по умолчанию ≈ 0.95rem = 15.2px на 16px root. */
  font-size: var(--code-font-size, 15px);
  line-height: 1.5;
  font-weight: 500;            /* чуть плотнее, лучше читается */
  border: none;
  border-radius: 0;
  resize: none;
  min-height: 0;
  caret-color: #f0f6fc !important;   /* ярче для модалки */
  outline: none;               /* убираем браузерный focus-ring */
  box-shadow: none;
}
.program-code--modal:focus,
.program-code--modal:focus-visible {
  outline: none;
  box-shadow: none;
  border: none;
}
.code-overlay--modal {
  font-size: var(--code-font-size, 15px);
  line-height: 1.5;
  font-weight: 500;
  color: #f0f6fc;
  background: #0d1117;
  border: none;
  border-radius: 0;
  padding: 0.6rem 0.8rem;
}
/* В модалке padding чуть больше — синхронизируем у textarea тоже */
.code-overlay--modal + .program-code--modal,
.program-code--modal {
  padding: 0.6rem 0.8rem;
}

/* ── Модал: код + журнал — вертикальный сплит ─────────── */
.code-wrap--modal {
  flex: 1 1 60%;
  display: flex;
  flex-direction: column;
  min-height: 0;     /* критично: без этого flex-child не сожмётся */
  background: #0d1117;
  overflow: hidden;
}
/* Редактор модала — textarea + overlay-подсветка. #cm-modal-fallback ===
   .code-wrap__editor (стили унифицированы ниже, row: gutter слева, редактор
   справа). Правило hidden — на случай программного скрытия контейнера. */
#cm-modal-fallback[hidden] { display: none; }
.code-wrap__editor {
  flex: 1;
  position: relative;
  display: flex;
  flex-direction: row;       /* gutter слева, редактор справа */
  min-width: 0;
  min-height: 0;
}
.code-wrap__editor-inner {
  flex: 1;
  position: relative;        /* для абсолютного overlay'а */
  min-width: 0;
  min-height: 0;
  display: flex;
  flex-direction: column;    /* textarea с flex:1 растягивается по ВЫСОТЕ */
  overflow: hidden;          /* textarea не должен выпадать за рамки */
}
/* Модальный гуттер: уравнять line-height с textarea, шрифт берётся
   из общей CSS-переменной (родительский .code-wrap--modal задаёт её
   через Shift+колесо мыши, см. attachCodeHighlight). */
.code-gutter--modal {
  line-height: 1.5;          /* совпадает с .program-code--modal:line-height */
  padding: 0.6rem 0.6rem 0.6rem 0.4rem;  /* по вертикали как у textarea (0.6rem) */
}
.code-gutter {
  flex: 0 0 auto;
  width: 3.5rem;
  padding: 0.6rem 0.5rem 0.6rem 0.3rem;
  background: #0a0e14;
  color: rgba(139, 148, 158, 0.55);
  font-family: "Consolas", "Menlo", monospace;
  font-size: var(--code-font-size, 15px);
  line-height: 1.5;
  text-align: right;
  white-space: pre;
  overflow: hidden;     /* скроллится в скрипте синхронно с textarea */
  user-select: none;
  border-right: 1px solid rgba(139, 148, 158, 0.15);
}
.code-modal__journal {
  flex: 1 1 40%;
  display: flex;
  flex-direction: column;
  border-top: 2px solid var(--border);
  background: #0d1117;
  min-height: 0;
}
.code-modal__journal-header {
  display: flex;
  justify-content: space-between;
  align-items: center;
  padding: 0.35rem 0.8rem;
  font-size: 0.75rem;
  color: var(--text-dim);
  text-transform: uppercase;
  letter-spacing: 0.05em;
  border-bottom: 1px solid rgba(139, 148, 158, 0.15);
}
.code-modal__hint {
  font-size: 0.7rem;
  color: rgba(139, 148, 158, 0.6);
  text-transform: none;
  letter-spacing: 0;
}
.code-modal__journal-body {
  flex: 1;
  overflow-y: auto;
  padding: 0.3rem 0.8rem;
  font-size: var(--journal-font-size, 14px);
  line-height: 1.25;        /* плотнее: было 1.4 */
  display: flex;
  flex-direction: column;
  gap: 0.1rem;              /* вдвое меньше: было 0.2rem */
}
.code-modal__journal-body .log-entry {
  font-size: inherit;       /* масштабируется через --journal-font-size */
  padding: 0.1rem 0.4rem;   /* вдвое меньше: было 0.25rem (вертикали) */
  line-height: 1.25;
}


/* ── Page header ─────────────────────────────────────────────────────────────── */
.page-header { margin-bottom: 1.5rem; }
.page-header h1 { font-size: 1.6rem; font-weight: 700; }
.subtitle { color: var(--text-dim); font-size: 0.9rem; margin-top: 0.25rem; }
/* Компактная версия — для страниц с большим контентом ниже. */
.page-header--compact { margin-bottom: 0.6rem; }
.page-header--compact h1 { font-size: 1.25rem; margin: 0; }
.page-header--compact .subtitle { font-size: 0.8rem; margin-top: 0.1rem; }

/* Заголовок-строка: подзаголовок слева, действия справа */
.page-header--row {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 1rem;
  flex-wrap: wrap;
}
.page-header--row > div { flex: 1; min-width: 240px; }
.page-header--row form  { flex-shrink: 0; }

/* ── Training layout ─────────────────────────────────────────────────────────── */
.training-layout { display: flex; flex-direction: column; gap: 1rem; max-width: 900px; }

.cmd-table { width: 100%; border-collapse: collapse; font-size: 0.85rem; }
.cmd-table th {
  text-align: left; padding: 0.4rem 0.6rem;
  background: var(--bg-panel); color: var(--text-dim);
  border-bottom: 1px solid var(--border);
}
.cmd-table td { padding: 0.35rem 0.6rem; border-bottom: 1px solid var(--border); }
.cmd-table code { background: var(--bg-panel); padding: 0.1rem 0.3rem; border-radius: 4px; color: var(--accent); }
.row--danger td  { background: rgba(248,81,73,0.05); }
.row--caution td { background: rgba(210,153,34,0.05); }

/* ── Missions ─────────────────────────────────────────────────────────────────── */
.missions-list { display: flex; flex-direction: column; gap: 0.75rem; }
.mission-card {
  padding: 0.75rem;
  background: var(--bg-panel);
  border-radius: var(--radius);
  border-left: 3px solid var(--border);
}
.mission-card--easy   { border-color: var(--success); }
.mission-card--medium { border-color: var(--warning); }
.mission-card--hard   { border-color: var(--danger); }
.mission-card__header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 0.4rem; }
.mission-card__title  { font-weight: 600; }
.mission-card__desc   { font-size: 0.82rem; color: var(--text-dim); margin-bottom: 0.4rem; }
.mission-card__meta   { display: flex; gap: 1rem; font-size: 0.75rem; color: var(--text-dim); margin-bottom: 0.5rem; }
.difficulty-badge { font-size: 0.7rem; padding: 0.1rem 0.4rem; border-radius: 99px; }
.difficulty-badge--easy   { background: #1a3322; color: var(--success); }
.difficulty-badge--medium { background: #3d2f00; color: var(--warning); }
.difficulty-badge--hard   { background: #3d1a1a; color: var(--danger); }

.steps-list { padding-left: 1.2rem; line-height: 2; font-size: 0.85rem; }
.note { margin-top: 0.75rem; font-size: 0.8rem; color: var(--text-dim); }
.note code { color: var(--accent); background: var(--bg-panel); padding: 0.1rem 0.3rem; border-radius: 3px; }

/* ── Sessions history ────────────────────────────────────────────────────────── */
.sessions-layout { display: flex; flex-direction: column; gap: 0.75rem; max-width: 700px; }
.session-card { background: var(--bg-card); border: 1px solid var(--border); border-radius: var(--radius); padding: 0.75rem; }
.session-card__header { display: flex; align-items: center; gap: 0.5rem; margin-bottom: 0.5rem; }
.session-card__id   { font-weight: 700; color: var(--accent); }
.session-card__date { font-size: 0.8rem; color: var(--text-dim); flex: 1; }
.session-card__body { display: flex; gap: 1rem; margin-bottom: 0.5rem; }
.session-stat { text-align: center; }
.session-stat__label { display: block; font-size: 0.65rem; color: var(--text-dim); }
.session-stat__value { font-size: 1.1rem; font-weight: 700; color: var(--text); }
.session-log summary { cursor: pointer; font-size: 0.78rem; color: var(--text-dim); }
.cmd-history { margin-top: 0.4rem; }
.cmd-entry { display: flex; gap: 0.5rem; font-size: 0.75rem; padding: 0.15rem 0; }
.cmd-entry--ok  { color: var(--success); }
.cmd-entry--err { color: var(--danger); }
.cmd-time   { color: var(--text-dim); min-width: 55px; }
.cmd-intent { min-width: 90px; font-weight: 600; }
.cmd-text   { color: var(--text-dim); }

.empty-hint { color: var(--text-dim); font-size: 0.85rem; }

/* ── Settings page ───────────────────────────────────────────────────────────── */
.settings-layout { max-width: 1150px; display: flex; flex-direction: column; gap: 1rem; padding: 1rem; }
/* Панели настроек — в 2 колонки (multi-column: карточки балансируются
   по высоте). На узком экране columns сам схлопнется в одну. */
.settings-form   { column-count: 2; column-gap: 1rem; }
.settings-form > .card { break-inside: avoid; margin-bottom: 1rem; }
.settings-form > .settings-actions { column-span: all; margin-bottom: 0; }
@media (max-width: 760px) {
  .settings-form { column-count: 1; }
}

.settings-grid {
  display: grid;
  grid-template-columns: 1fr 1fr;
  gap: 0.4rem 0.75rem;
  align-items: center;
}

.settings-label { font-size: 0.82rem; color: var(--text-dim); }
.settings-hint  { font-size: 0.72rem; color: var(--text-dim); grid-column: 2; margin-top: -0.2rem; }

.settings-input {
  background:  var(--bg-panel);
  border:      1px solid var(--border);
  border-radius: 4px;
  color:       var(--text);
  padding:     0.3rem 0.5rem;
  font-size:   0.85rem;
  width:       100%;
}
.settings-input:focus { outline: none; border-color: var(--accent); }

.settings-toggle { font-size: 0.82rem; }

.settings-actions { display: flex; gap: 0.75rem; align-items: center; padding-top: 0.25rem; }

/* ── Соединение с роботом: радио-карточки режима ──────────────────────────── */
.conn-modes { display: flex; flex-direction: column; gap: 0.4rem; margin-bottom: 0.3rem; }
.conn-mode {
  display: flex; align-items: flex-start; gap: 0.6rem;
  padding: 0.55rem 0.7rem;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 5px;
  cursor: pointer;
  transition: border-color .15s, background .15s;
}
.conn-mode:hover { border-color: var(--accent); }
.conn-mode input[type="radio"] { margin-top: 2px; flex-shrink: 0; cursor: pointer; }
.conn-mode:has(input:checked) {
  border-color: var(--accent);
  background: color-mix(in srgb, var(--accent) 8%, transparent);
}
.conn-mode__body { display: flex; flex-direction: column; gap: 0.15rem; font-size: 0.85rem; }
.conn-mode__hint { color: var(--text-dim); font-size: 0.78rem; line-height: 1.35; }

.conn-advanced {
  margin-top: 0.6rem;
  padding: 0.4rem 0.6rem;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 5px;
}
.conn-advanced > summary {
  cursor: pointer; font-size: 0.82rem; color: var(--text-dim);
  user-select: none;
}
.conn-advanced[open] > summary { color: var(--text); margin-bottom: 0.3rem; }

/* ── Scrollbar ───────────────────────────────────────────────────────────────── */
::-webkit-scrollbar       { width: 6px; }
::-webkit-scrollbar-track { background: transparent; }
::-webkit-scrollbar-thumb { background: var(--border); border-radius: 3px; }




/* ── Аутентификация ──────────────────────────────────────────────────────── */
.auth-layout {
  display: flex;
  align-items: center;
  justify-content: center;
  min-height: calc(100vh - 200px);
  padding: 2rem 1rem;
}
.auth-form {
  width: 100%;
  max-width: 360px;
  background: var(--bg-card, #161b22);
  border: 1px solid var(--border, #30363d);
  border-radius: 8px;
  padding: 1.75rem 1.5rem;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.auth-title { margin: 0 0 0.75rem; font-size: 1.25rem; }
.auth-error {
  padding: 0.6rem 0.8rem;
  background: rgba(248, 81, 73, 0.12);
  border: 1px solid rgba(248, 81, 73, 0.4);
  color: #f85149;
  border-radius: 6px;
  font-size: 0.9rem;
  margin-bottom: 0.5rem;
}
.auth-alt {
  margin-top: 1rem;
  text-align: center;
  font-size: 0.88rem;
  color: var(--muted, #8b949e);
}
.auth-alt a { color: var(--accent, #58a6ff); text-decoration: none; }
.auth-alt a:hover { text-decoration: underline; }

.topbar__user {
  display: flex;
  gap: 0.6rem;
  align-items: center;
  font-size: 0.85rem;
  color: var(--muted, #8b949e);
}
.topbar__user a {
  color: var(--accent, #58a6ff);
  text-decoration: none;
  padding: 0.25rem 0.6rem;
  border: 1px solid var(--border, #30363d);
  border-radius: 4px;
  font-size: 0.8rem;
}
.topbar__user a:hover { background: rgba(88, 166, 255, 0.1); }


/* ── Админ-панель ────────────────────────────────────────────────────────── */
.admin-layout {
  max-width: 1200px;
  margin: 0 auto;
  padding: 1.5rem 1rem;
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.admin-create-form {
  display: flex;
  gap: 0.5rem;
  align-items: center;
  flex-wrap: wrap;
}
.admin-create-form .settings-input { flex: 1 1 200px; }
.admin-table {
  width: 100%;
  border-collapse: collapse;
  font-size: 0.88rem;
}
.admin-table th, .admin-table td {
  padding: 0.5rem 0.6rem;
  border-bottom: 1px solid var(--border, #30363d);
  text-align: left;
  vertical-align: middle;
}
.admin-table th {
  background: rgba(88, 166, 255, 0.07);
  font-weight: 600;
  color: var(--muted, #8b949e);
}
.admin-table tbody tr:hover { background: rgba(88, 166, 255, 0.04); }
.admin-pwd {
  font-family: monospace;
  font-size: 1rem;
  color: #d29922;
  background: rgba(210, 153, 34, 0.1);
  padding: 0.15rem 0.4rem;
  border-radius: 3px;
}
.admin-password-cell { white-space: nowrap; }
.admin-actions { white-space: nowrap; display: flex; gap: 0.3rem; flex-wrap: wrap; align-items: center; }
.settings-input--inline { padding: 0.3rem 0.5rem; font-size: 0.85rem; max-width: 180px; }

.auth-success {
  padding: 0.6rem 0.8rem;
  background: rgba(63, 185, 80, 0.12);
  border: 1px solid rgba(63, 185, 80, 0.4);
  color: #3fb950;
  border-radius: 6px;
  font-size: 0.9rem;
  margin-bottom: 0.5rem;
}


/* ── Библиотека маршрутов ───────────────────────────────────────────────── */
.library-layout {
  max-width: 1100px;
  margin: 0 auto;
  padding: 1.5rem 1rem;
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.library-grid {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
  gap: 0.75rem;
}
.library-card {
  background: var(--bg-card, #161b22);
  border: 1px solid var(--border, #30363d);
  border-radius: 6px;
  padding: 0.85rem 1rem;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.library-card__title {
  font-size: 1rem;
  font-weight: 600;
}
.library-card__title a {
  color: var(--accent, #58a6ff);
  text-decoration: none;
}
.library-card__title a:hover { text-decoration: underline; }
.library-card__meta {
  display: flex;
  gap: 0.75rem;
  flex-wrap: wrap;
  font-size: 0.78rem;
  color: var(--muted, #8b949e);
}
.library-card__desc {
  font-size: 0.85rem;
  color: #c9d1d9;
  line-height: 1.4;
  max-height: 4.4em;
  overflow: hidden;
  text-overflow: ellipsis;
  display: -webkit-box;
  -webkit-line-clamp: 3;
  -webkit-box-orient: vertical;
}
.library-card__actions {
  display: flex;
  gap: 0.4rem;
  flex-wrap: wrap;
  align-items: center;
  margin-top: auto;
}
.library-detail__desc {
  margin-top: 1rem;
  white-space: pre-wrap;
  color: #c9d1d9;
  line-height: 1.5;
}
.library-detail__actions {
  display: flex;
  gap: 0.5rem;
  margin-top: 1rem;
  flex-wrap: wrap;
}
.library-code {
  background: #0d1117;
  border: 1px solid var(--border, #30363d);
  border-radius: 4px;
  padding: 0.75rem 1rem;
  overflow-x: auto;
  max-height: 600px;
  overflow-y: auto;
  font-family: monospace;
  font-size: 0.85rem;
  line-height: 1.45;
  white-space: pre;
  color: #c9d1d9;
}
.library-code code { background: transparent; padding: 0; }


/* ── Двухколоночная библиотека ──────────────────────────────────────────── */
.library-columns {
  display: grid;
  grid-template-columns: 1fr 1fr;
  gap: 1rem;
}
.library-list {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  margin-top: 0.5rem;
  /* У каждой колонки свой независимый скролл карточек. */
  max-height: 60vh;
  overflow-y: auto;
  padding-right: 0.3rem;
}
@media (max-width: 900px) {
  .library-columns { grid-template-columns: 1fr; }
}


/* ── Компас 3×3 ───────────────────────────────────────────────────────── */
.compass-grid {
  display: grid;
  grid-template-columns: 1fr 1fr 1fr;
  gap: 0.25rem;
  max-width: 200px;
  margin-top: 0.25rem;
}
.compass-grid .btn { padding: 0.3rem 0; font-size: 0.78rem; }
.compass-grid__center {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  font-size: 1.1rem;
  color: var(--muted, #8b949e);
  user-select: none;
}

/* ════════════════════════════════════════════════════════════════════
   Страница «🎯 Миссии» (/tasks)
   ════════════════════════════════════════════════════════════════════ */

.missions-page { display: flex; flex-direction: column; gap: 1rem; }

/* ── Табы ─────────────────────────────────────────────────────────── */
.tabs {
  display: flex; gap: 0.4rem;
  border-bottom: 1px solid var(--border);
  padding: 0 0.2rem;
}
.tabs__btn {
  background: transparent; color: var(--text-dim);
  border: 1px solid transparent; border-bottom: none;
  padding: 0.55rem 1rem;
  font-size: 0.9rem; font-weight: 500;
  cursor: pointer; font-family: inherit;
  border-radius: 6px 6px 0 0;
  margin-bottom: -1px;            /* чтобы активная заехала на border */
  display: inline-flex; align-items: center; gap: 0.4rem;
  transition: background 0.15s, color 0.15s, border-color 0.15s;
}
.tabs__btn:hover { background: rgba(255,255,255,0.04); color: var(--text); }
.tabs__btn--active {
  color: var(--accent);
  background: var(--bg-card);
  border-color: var(--border);
  border-bottom-color: var(--bg-card);
  font-weight: 600;
}
.tabs__count {
  font-size: 0.7rem; padding: 1px 7px; border-radius: 10px;
  background: var(--border); color: var(--text-dim);
}
.tabs__btn--active .tabs__count { background: var(--accent); color: #fff; }

.tab-panel { display: none; }
.tab-panel--active { display: block; }
/* Принудительное скрытие активного таба, когда показывается превью.
   Использует !important чтобы перебить .tab-panel--active. */
.tab-panel--hidden { display: none !important; }

/* ── Сетка карточек уровней (компактная) ──────────────────────────── */
.level-grid {
  display: grid; gap: 0.5rem;
  grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
  margin-top: 0.5rem;
}
.level-card {
  display: block; cursor: pointer;
  background: var(--bg-panel);
  border: 1px solid var(--border);
  border-radius: 6px;
  padding: 0.5rem 0.65rem;
  transition: border-color 0.15s, background 0.15s;
}
.level-card input[type=radio] { display: none; }
.level-card:hover { border-color: var(--text-dim); }
.level-card--selected {
  border-color: var(--accent);
  background: rgba(88,166,255,0.06);
}
.level-card__head {
  display: flex; align-items: center; gap: 0.45rem;
  margin-bottom: 0.25rem;
}
.level-card__num {
  display: inline-flex; align-items: center; justify-content: center;
  width: 24px; height: 24px;
  border: 1px solid transparent; border-radius: 50%;
  font-weight: 700; font-size: 0.8rem; color: #fff;
  flex-shrink: 0;
}
.level-card__num--1 { background: #2ea043; }
.level-card__num--2 { background: #58a6ff; }
.level-card__num--3 { background: #d4a017; }
.level-card__num--4 { background: #d28438; }
.level-card__num--5 { background: #d2392f; }
.level-card--selected .level-card__num {
  box-shadow: 0 0 0 2px var(--bg-card), 0 0 0 3px var(--accent);
}
.level-card__name { font-size: 1.05rem; font-weight: 600; color: var(--text); }
.level-card__desc { font-size: 0.75rem; color: var(--text-dim);
                    line-height: 1.4; margin: 0; }

.level-card__criteria { font-size: 0.75rem; color: var(--text-dim);
                    line-height: 1.4; margin: 0.45rem 0 0; }

.generate-actions {
  margin-top: 0.6rem; padding-top: 0.5rem;
  border-top: 1px solid var(--border);
  display: flex; align-items: center;
}

/* Бейдж id миссии в превью */
.preview-id-chip {
  font-family: monospace; font-size: 0.75rem;
  color: var(--text-dim); padding: 1px 6px;
  background: var(--bg-panel); border-radius: 4px;
  margin-left: 0.4rem;
}

/* (Раньше здесь был .toast-flash — заменён на использование
   статус-строки подвала #status-bar через flash() в tasks.html.) */

/* ── Превью миссии (на весь доступный экран) ──────────────────────── */
/* Карточка превью растягивается до низа viewport. Уменьшение level-
   селектора при открытии превью освобождает вертикаль для большого
   SVG-превью + описания. */
.preview-card {
  padding: 0.7rem 0.9rem;
  display: flex; flex-direction: column;
  /* Topbar (3.5rem) + page-header (~3rem) + tabs (~2.5rem) + appbar
     (22px) + paddings — резерв около 13rem из 100vh. */
  min-height: calc(100vh - 13rem);
}
.preview-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  gap: 1rem; margin-top: 0.5rem;
  flex: 1; min-height: 0;
}
@media (max-width: 760px) {
  .preview-grid { grid-template-columns: 1fr; }
}
.preview-svg-wrap {
  background: #0d1117; border: 1px solid var(--border);
  border-radius: 6px; padding: 0.4rem;
  display: flex; align-items: center; justify-content: center;
  /* Делаем максимальный квадрат, который влезает в ячейку грида:
     aspect-ratio + max-width/max-height с margin:auto. */
  aspect-ratio: 1;
  max-width: 100%; max-height: 100%;
  margin: 0 auto;
}
.preview-svg { width: 100%; height: 100%; display: block; }
.preview-text {
  display: flex; flex-direction: column; gap: 0.5rem;
  min-height: 0; min-width: 0;
}
.preview-text h3 { margin: 0; font-size: 1.05rem; color: var(--text); }
/* Редактируемое название миссии в превью. Стилизовано как заголовок,
   но позволяет вводить текст. После сохранения становится readonly. */
.preview-title-input {
  font-size: 1.15rem; font-weight: 700; color: var(--text);
  background: transparent;
  border: 1px dashed var(--border); border-radius: 5px;
  padding: 0.4rem 0.6rem;
  font-family: inherit; width: 100%;
}
.preview-title-input:focus {
  outline: none; border-color: var(--accent); border-style: solid;
  background: var(--bg-panel);
}
.preview-title-input:read-only {
  border-color: transparent; padding-left: 0;
  background: transparent; cursor: default;
}
.preview-title-input::placeholder { color: var(--text-dim); font-weight: 500; }
.preview-desc {
  margin: 0; padding: 0.7rem 0.9rem;
  background: var(--bg-panel); border: 1px solid var(--border);
  border-radius: 5px;
  font-family: inherit; font-size: 0.92rem; color: var(--text);
  white-space: pre-wrap; line-height: 1.55;
  flex: 1; overflow-y: auto; min-height: 0;
}

/* ── Сетка карточек миссий (Опубликованные / Мои) ─────────────────── */
.missions-grid {
  display: grid; gap: 0.8rem;
  grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
}
.mission-card {
  background: var(--bg-card); border: 1px solid var(--border);
  border-radius: 8px; padding: 0.9rem;
  display: flex; flex-direction: column; gap: 0.5rem;
  transition: border-color 0.15s, transform 0.15s;
}
.mission-card:hover {
  border-color: var(--accent);
  transform: translateY(-1px);
}
.mission-card__head { display: flex; align-items: center; gap: 0.5rem; }
.mission-card__title {
  margin: 0; font-size: 0.95rem; font-weight: 600;
  color: var(--text); flex: 1;
}
.mission-card__owner { margin: 0; font-size: 0.78rem; color: var(--text-dim); }
.mission-card__desc {
  margin: 0; font-size: 0.82rem; color: var(--text-dim);
  line-height: 1.45; flex: 1;
  white-space: pre-line;   /* переносы по \n из описания миссии */
}
/* «Набрано: N⭐» — лучший результат пользователя по миссии. Показывается
   только при наличии завершённых прохождений. */
.mission-card__score {
  margin: 0; font-size: 0.82rem; color: var(--text-dim);
  padding: 0.25rem 0.5rem;
  background: rgba(245,184,40,0.08);
  border: 1px solid rgba(245,184,40,0.25);
  border-radius: 4px;
}
.mission-card__score strong { color: var(--warning); font-weight: 600; }
.mission-card__actions {
  display: flex; gap: 0.4rem; flex-wrap: wrap;
  padding-top: 0.4rem; border-top: 1px solid var(--border);
}

/* Цветные бейджи уровня */
.mission-card__level {
  display: inline-flex; align-items: center; justify-content: center;
  width: 24px; height: 24px;
  border-radius: 50%; font-size: 0.78rem; font-weight: 700;
  color: #fff;
}
.mission-card__level--0 { background: #8b5cf6; }   /* фиолетовый = Кастомная */
.mission-card__level--1 { background: #2ea043; }   /* зелёный */
.mission-card__level--2 { background: #58a6ff; }   /* синий */
.mission-card__level--3 { background: #d4a017; }   /* жёлтый */
.mission-card__level--4 { background: #d28438; }   /* оранжевый */
.mission-card__level--5 { background: #d2392f; }   /* красный */
/* ID-чип в правом верхнем углу карточки */
.mission-card__id {
  font-family: monospace; font-size: 0.7rem; color: var(--text-dim);
  padding: 1px 5px; background: var(--bg-panel); border-radius: 4px;
  margin-left: auto;
}
/* Бейдж «опубликована» возле текста владельца */
.mission-card__pubbadge {
  font-size: 0.7rem; color: var(--success);
  margin-left: 0.4rem; padding: 1px 6px;
  background: rgba(46,160,67,0.10); border-radius: 4px;
}

/* ── Каталог: тулбар с фильтром и поиском ────────────────────────── */
.catalog-toolbar {
  display: flex; flex-wrap: wrap; gap: 0.6rem;
  align-items: center; justify-content: space-between;
  margin-bottom: 0.8rem;
}
.filter-chips { display: inline-flex; gap: 0.35rem; flex-wrap: wrap; }
.chip {
  display: inline-flex; align-items: center; gap: 0.35rem;
  padding: 0.32rem 0.7rem; font-size: 0.8rem; font-weight: 500;
  background: var(--bg-panel); color: var(--text-dim);
  border: 1px solid var(--border); border-radius: 14px;
  cursor: pointer; font-family: inherit;
  transition: background 0.15s, color 0.15s, border-color 0.15s;
}
.chip:hover { color: var(--text); border-color: var(--text-dim); }
.chip--active {
  background: rgba(88,166,255,0.10);
  color: var(--accent);
  border-color: var(--accent);
  font-weight: 600;
}
.chip__count {
  font-size: 0.68rem; padding: 0 5px; border-radius: 8px;
  background: var(--border); color: var(--text-dim);
  min-width: 18px; text-align: center;
}
.chip--active .chip__count { background: var(--accent); color: #fff; }

.catalog-search { flex: 1; min-width: 200px; max-width: 360px; }
.catalog-search .settings-input {
  width: 100%; padding: 0.4rem 0.7rem;
  font-size: 0.85rem;
}

/* Второй ряд chips — фильтр по уровню сложности. */
.filter-chips--level { margin-bottom: 0.8rem; }
.chip--lvl { padding-left: 0.45rem; }
.chip--lvl .chip__lvlnum {
  display: inline-flex; align-items: center; justify-content: center;
  width: 18px; height: 18px;
  border-radius: 50%; font-weight: 700; font-size: 0.7rem; color: #fff;
}
.chip--lvl-0 .chip__lvlnum { background: #8b5cf6; font-size: 0.65rem; }

/* ── Sticky-кнопка активной миссии в правой панели (журнал) ─────────── */
.mission-show-btn {
  display: flex; align-items: center; gap: 0.4rem;
  margin: 0 0 0.5rem 0;
  padding: 0.55rem 0.8rem;
  background: rgba(88,166,255,0.10);
  border: 1px solid rgba(88,166,255,0.35);
  color: var(--text);
  font-size: 0.85rem;
  width: 100%;
  text-align: left;
  flex-wrap: wrap;
}
.mission-show-btn:hover { background: rgba(88,166,255,0.16); }
.mission-show-btn strong { font-weight: 400; flex: 1; min-width: 0; overflow: hidden; text-overflow: ellipsis; }
.mission-show-btn__chip {
  font-size: 0.7rem; padding: 2px 7px;
  background: var(--bg-panel); border-radius: 10px;
  color: var(--text-dim); font-weight: 600;
  white-space: nowrap;
}
.mission-show-btn__chip--warn { color: var(--danger); }

/* ── Иконка миссии-результата в модалке ─────────────────────────────── */
.mission-result-icon { font-size: 1.4rem; }
.mission-result-body { font-size: 0.95rem; line-height: 1.6; }

/* ── Таймер активной миссии в шапке журнала ────────────────────────── */
.mission-timer {
  font-family: ui-monospace, "JetBrains Mono", Consolas, monospace;
  font-size: 0.78rem;
  padding: 0.2rem 0.5rem;
  border-radius: 4px;
  background: #14202f;
  color: var(--accent);
  border: 1px solid #1e2c40;
  white-space: nowrap;
}

/* ── Блокировка кнопок режима миссией с опасными зонами ─────────────────
   .is-locked + disabled — UX-индикация «менять режим нельзя». Серверная
   валидация (mode_inspector/mode_cautious) в любом случае отвергает
   переключение, это просто визуальная подсказка. */
#btn-mode-inspector.is-locked,
#btn-mode-cautious.is-locked,
#btn-goto.is-locked,
#btn-set-zone.is-locked,
#btn-autopilot.is-locked,
#btn-zone-mode.is-locked {
  opacity:        0.55;
  cursor:         not-allowed;
  filter:         grayscale(35%);
}

/* ── Action-кнопки: цвет = активный режим робота ────────────────────────
   .btn--mode-action и .btn--voice окрашиваются в палитру активной
   кнопки текущего режима, чтобы запуск/слушание всегда показывали,
   в каком режиме сейчас работает робот:
     • инспектор (по умолчанию) → насыщенный синий (как .btn--info.is-active)
     • осторожно (body.mode-cautious) → насыщенный жёлтый (как .btn--warn.is-active)
*/
.btn--mode-action {
  background:   #1f4068 !important;
  color:        #fff !important;
  border-color: var(--accent) !important;
}
.btn--mode-action:hover {
  background:   #2a5180 !important;
  border-color: var(--accent) !important;
}

body.mode-cautious .btn--mode-action {
  background:   #6b4200 !important;
  color:        #fff !important;
  border-color: var(--warning) !important;
}
body.mode-cautious .btn--mode-action:hover {
  background:   #8a5500 !important;
  border-color: var(--warning) !important;
}

/* Кнопка «🎙 Слушать» — особая (width:100%, состояние .listening).
   Палитра та же: режим = цвет, listening = более насыщенный оттенок. */
.btn--voice {
  background:   #1f4068 !important;
  color:        #fff !important;
  border-color: var(--accent) !important;
}
.btn--voice:hover    { background: #2a5180 !important; }
.btn--voice.listening {
  background:   #2a5180 !important;
  border-color: var(--accent2) !important;
}

body.mode-cautious .btn--voice {
  background:   #6b4200 !important;
  color:        #fff !important;
  border-color: var(--warning) !important;
}
body.mode-cautious .btn--voice:hover    { background: #8a5500 !important; }
body.mode-cautious .btn--voice.listening {
  background:   #8a5500 !important;
  border-color: #ffd166 !important;
}

body.mode-cautious .voice-status {
  color: var(--warning) !important;
  border-top-color: rgba(245,184,40,0.25) !important;
}

/* ── Переключатель публикации миссии (toggle-switch) ───────────────── */
/* Использует hidden checkbox + кастомный slider, чтобы видеть состояние
   мгновенно без перезагрузки. Сидит в .mission-card__actions рядом с
   обычными .btn-кнопками — поэтому выровнен по геометрии. */
.pub-switch {
  display: inline-flex; align-items: center; gap: 0.45rem;
  padding: 0.45rem 0.7rem;
  border: 1px solid var(--border); border-radius: 6px;
  background: var(--bg-panel); cursor: pointer;
  font-size: 0.8rem; color: var(--text-dim);
  user-select: none;
  position: relative;                     /* анкор для absolute input */
  transition: background 0.15s, border-color 0.15s, color 0.15s;
}
.pub-switch:hover { background: rgba(255,255,255,0.04); }
/* «Visually hidden but accessible» — стандартный паттерн.
   НЕ используем pointer-events:none, иначе label-click-toggle ломается. */
.pub-switch input[type=checkbox] {
  position: absolute;
  width: 1px; height: 1px;
  padding: 0; margin: -1px;
  overflow: hidden;
  clip: rect(0, 0, 0, 0);
  white-space: nowrap;
  border: 0;
}
.pub-switch__slider {
  position: relative;
  width: 28px; height: 16px;
  background: #30363d; border-radius: 10px;
  transition: background 0.15s;
  flex-shrink: 0;
}
.pub-switch__slider::before {
  content: ''; position: absolute;
  width: 12px; height: 12px; top: 2px; left: 2px;
  background: #c9d1d9; border-radius: 50%;
  transition: transform 0.18s;
}
.pub-switch input:checked + .pub-switch__slider {
  background: var(--success);
}
.pub-switch input:checked + .pub-switch__slider::before {
  transform: translateX(12px);
  background: #fff;
}
.pub-switch input:checked ~ .pub-switch__txt {
  color: var(--success); font-weight: 600;
}
.chip--lvl-1 .chip__lvlnum { background: #2ea043; }
.chip--lvl-2 .chip__lvlnum { background: #58a6ff; }
.chip--lvl-3 .chip__lvlnum { background: #d4a017; }
.chip--lvl-4 .chip__lvlnum { background: #d28438; }
.chip--lvl-5 .chip__lvlnum { background: #d2392f; }

/* ── Popup-автодополнение robot.X для textarea ─────────────────────────
   Структура: popup = scrollable __list + фиксированная __doc-полоса
   снизу с описанием выбранного пункта. Строки списка ОДНОЙ высоты,
   стрелками вверх/вниз → меняется только подсветка и текст __doc,
   список не перерисовывается (нет прыжков). */
.robot-autocomplete {
  position: fixed;
  z-index: 10000;
  max-height: 320px;
  width: 420px;
  background: #1f2733;
  border: 1px solid #3a4654;
  border-radius: 4px;
  box-shadow: 0 6px 22px rgba(0, 0, 0, 0.55);
  font-family: "Consolas", "Menlo", monospace;
  font-size: 13px;
  color: #d8dde4;
  display: flex;
  flex-direction: column;
  overflow: hidden;
}
.robot-autocomplete[hidden] { display: none; }
.robot-autocomplete__list {
  flex: 1 1 auto;
  min-height: 0;
  overflow-y: auto;
  padding: 2px 0;
}
.robot-autocomplete__item {
  display: flex;
  align-items: baseline;
  gap: 6px;
  padding: 3px 10px;
  cursor: pointer;
  border-left: 3px solid transparent;
  line-height: 1.4;
  white-space: nowrap;
  overflow: hidden;
}
.robot-autocomplete__item--sel {
  background: rgba(88, 166, 255, 0.18);
  border-left-color: #58a6ff;
}
.robot-autocomplete__icon {
  display: inline-block;
  width: 1.1em;
  text-align: center;
  font-weight: 700;
  flex: 0 0 auto;
}
.robot-autocomplete__icon--method   { color: #b392f0; }   /* фиолетовый */
.robot-autocomplete__icon--property { color: #79c0ff; }   /* голубой */
.robot-autocomplete__name {
  color: #e6edf3;
  font-weight: 600;
  flex: 0 0 auto;
}
.robot-autocomplete__sig {
  color: #8b949e;
  font-size: 12px;
  flex: 1 1 auto;
  min-width: 0;
  overflow: hidden;
  text-overflow: ellipsis;
}
/* Фиксированная полоса с описанием выбранного метода. Высота стабильна:
   max 2 строки + ellipsis. min-height гарантирует одну строку даже у
   пустого описания — popup не «прыгает» по высоте при наведении на
   разные пункты. */
.robot-autocomplete__doc {
  flex: 0 0 auto;
  border-top: 1px solid rgba(139, 148, 158, 0.2);
  padding: 6px 10px;
  font-size: 11.5px;
  color: #9aa5b1;
  background: rgba(13, 17, 23, 0.55);
  line-height: 1.4;
  white-space: normal;
  min-height: 2.4em;         /* стабильная высота, без скачков */
  max-height: 4.2em;
  overflow-y: auto;
}
