/* ============================================================
   Bond Robot — application styles
   Drill Kit tokens (dark) + app-specific patterns
   ============================================================ */

* { box-sizing: border-box; }
html, body, #root { height: 100%; }
body {
  margin: 0;
  background: var(--bg-app);
  color: var(--text-primary);
  font-family: var(--font-sans);
  overflow: hidden;
  font-size: 13px;
  line-height: 1.45;
}
button { font-family: inherit; -webkit-appearance: none; appearance: none; }

/* App shell */
.shell { display: flex; height: 100vh; height: 100dvh; min-height: 0; }
.main  { flex: 1; display: flex; flex-direction: column; min-width: 0; }
.scroll { flex: 1; overflow: auto; min-height: 0; -webkit-overflow-scrolling: touch; }
.scroll::-webkit-scrollbar { width: 10px; height: 10px; }
.scroll::-webkit-scrollbar-thumb { background: var(--color-neutral-800); border-radius: 999px; }
.scroll::-webkit-scrollbar-track { background: transparent; }

/* Typography */
.mono { font-family: var(--font-mono); font-feature-settings: "tnum" 1, "lnum" 1; font-variant-numeric: tabular-nums; }
.num  { font-family: var(--font-mono); font-feature-settings: "tnum" 1, "lnum" 1; font-variant-numeric: tabular-nums; }
.eyebrow {
  font-family: var(--font-mono);
  font-size: 10px;
  text-transform: uppercase;
  letter-spacing: .06em;
  color: var(--text-tertiary);
}
.h1 { font-size: 22px; font-weight: 600; margin: 0; letter-spacing: -0.01em; white-space: nowrap; }
.h2 { font-size: 16px; font-weight: 600; margin: 0; white-space: nowrap; }
.h3 { font-size: 13px; font-weight: 600; margin: 0; white-space: nowrap; }
.eyebrow { white-space: nowrap; }
.label { font-size: 11px; color: var(--text-tertiary); text-transform: uppercase; letter-spacing: .04em; font-family: var(--font-mono); }
.sect-title { font-family: var(--font-mono); font-size: 10px; text-transform: uppercase; letter-spacing: .08em; color: var(--text-tertiary); }
.muted { color: var(--text-tertiary); }
.sec   { color: var(--text-secondary); }
.pri   { color: var(--text-primary); }
.brand { color: var(--text-brand); }
.pos   { color: var(--text-success); }
.neg   { color: var(--text-error); }
.warn  { color: var(--text-warning); }
.info  { color: var(--text-info); }

/* Card / Block */
.block {
  background: var(--bg-container);
  border: 1px solid var(--border-default);
  border-radius: 8px;
  display: flex;
  flex-direction: column;
  min-width: 0;
}
.block-head {
  display: flex;
  align-items: center;
  justify-content: space-between;
  padding: 12px 16px;
  border-bottom: 1px solid var(--border-default);
  gap: 12px;
  min-height: 48px;
}
.block-head-title { font-size: 13px; font-weight: 600; color: var(--text-primary); display: flex; align-items: center; gap: 8px; min-width: 0; }
/* Колонка eyebrow+title в шапке блока должна уметь сжиматься (иначе длинный eyebrow с
   white-space:nowrap распирает шапку вправо за вьюпорт на узких экранах). На десктопе места
   хватает — колонка не сжимается, вид не меняется. */
.block-head > .col { min-width: 0; }
.block-body { padding: 16px; }
.block-body.flush { padding: 0; }

/* Buttons */
.btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 6px;
  padding: 6px 12px;
  border: 1px solid var(--border-default);
  background: var(--bg-neutral);
  color: var(--text-primary);
  border-radius: var(--radius-button-md);
  font-size: 12px;
  font-weight: 500;
  cursor: pointer;
  font-family: inherit;
  transition: background .15s, border-color .15s, box-shadow .15s, transform .06s;
  white-space: nowrap;
}
.btn:hover { background: var(--bg-neutral-hover); border-color: var(--border-strong); }
.btn:active { background: var(--bg-neutral-press); transform: translateY(1px); }
.btn:focus-visible { outline: 2px solid var(--border-focus); outline-offset: 2px; }
.btn.primary {
  background: var(--bg-brand);
  border-color: var(--bg-brand);
  color: var(--text-on-brand);
  box-shadow: 0 1px 2px oklch(from var(--color-brand-500) l c h / 0.30);
}
.btn.primary:hover { background: var(--bg-brand-hover); border-color: var(--bg-brand-hover); box-shadow: 0 3px 10px oklch(from var(--color-brand-500) l c h / 0.26); }
.btn.primary:active { background: var(--bg-brand-press); border-color: var(--bg-brand-press); box-shadow: none; }
.btn:disabled,
.btn:disabled:hover {
  opacity: .55;
  cursor: not-allowed;
  box-shadow: none;
  transform: none;
  background: var(--bg-neutral);
  border-color: var(--border-default);
  color: var(--text-tertiary);
}
.btn.primary:disabled,
.btn.primary:disabled:hover {
  opacity: 1;
  background: var(--bg-brand-disabled);
  border-color: var(--bg-brand-disabled);
  color: var(--text-disabled);
}
/* Видимость: в dark-теме --border-default сливается с фоном контейнера,
   ghost-кнопки выглядели «голым текстом». border-strong отличим в обеих темах. */
.btn.ghost { background: transparent; border-color: var(--border-strong); }
.btn.ghost:hover { background: var(--bg-neutral); }
.btn.ghost:active { background: var(--bg-neutral-press); }
.btn.danger { color: var(--text-error); border-color: var(--color-error-800); }
.btn.danger:hover { background: var(--bg-error-tint); }
.btn.danger:active { background: var(--bg-error-tint); border-color: var(--border-error); }
.btn.sm { padding: 4px 8px; font-size: 11px; border-radius: var(--radius-button-sm); }
.btn.icon { padding: 6px; }
/* Состояние загрузки: текст прячется, крутится спиннер, повторный клик заблокирован.
   Навешивать класс loading на кнопку на время async-операции (Сохранить/Применить/Прогон). */
.btn.loading { color: transparent !important; pointer-events: none; position: relative; }
.btn.loading::after {
  content: ""; position: absolute; width: 13px; height: 13px; border-radius: 50%;
  border: 2px solid currentColor; border-top-color: transparent;
  color: var(--text-primary); animation: btnSpin .6s linear infinite;
}
.btn.primary.loading::after { color: var(--text-on-brand); }
.btn.danger.loading::after { color: var(--text-error); }
@keyframes btnSpin { to { transform: rotate(360deg); } }

/* Tag / pill */
.tag {
  display: inline-flex;
  align-items: center;
  gap: 4px;
  padding: 2px 8px;
  border-radius: 999px;
  background: var(--bg-neutral);
  border: 1px solid var(--border-default);
  font-size: 11px;
  color: var(--text-secondary);
  white-space: nowrap;
  font-family: var(--font-mono);
}
.tag.brand   { background: var(--bg-brand-tint);   color: var(--text-brand);   border-color: transparent; }
.tag.success { background: var(--bg-success-tint); color: var(--text-success); border-color: transparent; }
.tag.warning { background: var(--bg-warning-tint); color: var(--text-warning); border-color: transparent; }
.tag.error   { background: var(--bg-error-tint);   color: var(--text-error);   border-color: transparent; }
.tag.info    { background: var(--bg-info-tint);    color: var(--text-info);    border-color: transparent; }
.tag.muted   { background: var(--bg-neutral-hover); color: var(--text-tertiary); border-color: transparent; }
.tag.solid-brand   { background: var(--bg-brand);   color: var(--text-on-brand);  border-color: transparent; }

/* Status dot */
.dot { width: 8px; height: 8px; border-radius: 50%; flex-shrink: 0; display: inline-block; }
.dot.pos  { background: var(--color-success-500); box-shadow: 0 0 0 3px oklch(from var(--color-success-500) l c h / 0.18); }
.dot.warn { background: var(--color-warning-500); box-shadow: 0 0 0 3px oklch(from var(--color-warning-500) l c h / 0.18); }
.dot.neg  { background: var(--color-error-500);   box-shadow: 0 0 0 3px oklch(from var(--color-error-500)   l c h / 0.18); }
.dot.info { background: var(--color-info-500);    box-shadow: 0 0 0 3px oklch(from var(--color-info-500)    l c h / 0.18); }
.dot.off  { background: var(--color-neutral-600); }

/* Tabs */
.tabs { display: flex; gap: 0; border-bottom: 1px solid var(--border-default); }
.tab {
  padding: 10px 14px;
  font-size: 13px;
  background: transparent;
  border: 0;
  color: var(--text-secondary);
  cursor: pointer;
  font-weight: 500;
  border-bottom: 2px solid transparent;
  margin-bottom: -1px;
  font-family: inherit;
  display: flex;
  align-items: center;
  gap: 6px;
}
.tab:hover { color: var(--text-primary); }
.tab.active { color: var(--text-brand); border-bottom-color: var(--border-brand); }
.tab .count { font-family: var(--font-mono); font-size: 10px; padding: 1px 6px; border-radius: 999px; background: var(--bg-neutral); color: var(--text-tertiary); }
.tab.active .count { background: var(--bg-brand-tint); color: var(--text-brand); }

/* Tables */
table.t { width: 100%; border-collapse: collapse; font-size: 12px; }
table.t thead th {
  text-align: left;
  font-weight: 500;
  font-size: 10px;
  text-transform: uppercase;
  letter-spacing: .04em;
  color: var(--text-tertiary);
  padding: 10px 12px;
  border-bottom: 1px solid var(--border-default);
  background: var(--bg-container);
  position: sticky; top: 0; z-index: 1;
  font-family: var(--font-mono);
  white-space: nowrap;
}
table.t tbody td {
  padding: 10px 12px;
  border-bottom: 1px solid var(--border-default);
  vertical-align: middle;
  color: var(--text-secondary);
}
table.t tbody tr { cursor: pointer; transition: background .12s; }
table.t tbody tr:hover { background: var(--bg-neutral); }
table.t tbody tr.active { background: var(--bg-brand-tint); }
table.t tbody tr.active td { color: var(--text-primary); }
/* Карточка P&L «Обзора»: строки информативные, не кликабельные — убираем pointer/hover. */
table.t.pnl-flat tbody tr { cursor: default; }
table.t.pnl-flat tbody tr:hover { background: transparent; }
/* Листаем прошлый месяц → колонка «За сегодня» (2-я) к нему не относится — приглушаем. */
table.t.pnl-past tbody td:nth-child(2), table.t.pnl-past thead th:nth-child(2) { opacity: .4; }
table.t tbody tr.row-new {
  animation: row-new-flash 1.45s ease-out;
}
table.t tbody tr.row-new td {
  color: var(--text-primary);
}
table.t td.num, table.t th.num { font-family: var(--font-mono); text-align: right; }
/* Таблица «Торговля»: 15 столбцов (4 доходности + Погаш./Оферта). Числовые держим
   компактными и узкими, чтобы строка не распухала; заголовки доходностей не переносим. */
table.t.bonds-trade-table th.num,
table.t.bonds-trade-table td.num {
  white-space: nowrap;
  padding-left: 8px;
  padding-right: 8px;
}
/* Цифры торговой таблицы (цена, доходности, канал…) — чуть мельче и пожирнее:
   плотнее и контрастнее в густой 15-колоночной строке. Montserrat + табличные цифры. */
table.t.bonds-trade-table td.num { font-size: 11px; font-weight: 600; }
.group-title-edit {
  display: flex;
  align-items: center;
  gap: 8px;
  min-height: 36px;
}
.group-title-edit .h1 {
  margin: 0;
  cursor: text;
}
.group-title-input {
  width: min(560px, 52vw);
  height: 38px;
  font-size: 26px;
  font-weight: 800;
  letter-spacing: 0;
  color: var(--text-primary);
}
.sort-th {
  display: inline-flex;
  align-items: center;
  justify-content: inherit;
  gap: 5px;
  width: 100%;
  padding: 0;
  border: 0;
  background: transparent;
  color: inherit;
  font: inherit;
  text-transform: inherit;
  letter-spacing: inherit;
  cursor: pointer;
}
.sort-th:hover,
.sort-th.active {
  color: var(--text-primary);
}
th.num .sort-th {
  justify-content: flex-end;
}
.sort-th-arrow {
  color: var(--text-tertiary);
  font-size: 9px;
  line-height: 1;
}
.sort-th.active .sort-th-arrow {
  color: var(--text-brand);
}

/* Inputs */
.input, .select {
  background: var(--bg-container);
  border: 1px solid var(--border-default);
  border-radius: var(--radius-input);
  padding: 6px 10px;
  color: var(--text-primary);
  font-family: inherit;
  font-size: 12px;
  width: 100%;
  outline: none;
  transition: border-color .15s, box-shadow .15s;
}
.input:focus, .select:focus { border-color: var(--input-focus-color); box-shadow: 0 0 0 3px oklch(from var(--input-focus-color) l c h / 0.20); }
.select.sm { width: auto; padding: 4px 8px; font-size: 11px; }
.input::placeholder { color: var(--text-placeholder); }
.field { display: flex; flex-direction: column; gap: 4px; min-width: 0; }
.field-label { font-size: 10px; text-transform: uppercase; letter-spacing: .04em; color: var(--text-tertiary); font-family: var(--font-mono); }
.field-hint  { font-size: 11px; color: var(--text-tertiary); }
/* PROF-1: «тихие часы» — пара селектов ЧЧ : ММ в строку вместо нативного time-инпута */
.quiet-time-select { display: flex; align-items: center; gap: 6px; min-width: 0; }
.quiet-time-select > .input { flex: 1 1 0; min-width: 0; text-align: center; }
.quiet-time-select .quiet-time-sep { color: var(--text-tertiary); font-weight: 600; flex: 0 0 auto; }
.compact-field {
  display: grid;
  grid-template-columns: minmax(104px, 1fr) minmax(96px, 44%);
  align-items: center;
  gap: 8px;
  padding: 5px 0;
  border-bottom: 1px solid var(--border-default);
}
.compact-field .field-label {
  white-space: normal;
  line-height: 1.2;
}
.compact-field .input,
.compact-field .select {
  min-height: 28px;
  padding: 4px 8px;
  text-align: right;
}
.settings-field {
  gap: 5px;
  padding: 7px 0;
  border-bottom: 1px solid var(--border-default);
}
.settings-field .field-label {
  white-space: normal;
  line-height: 1.2;
}
.settings-field .input,
.settings-field .select {
  min-height: 30px;
  padding: 5px 8px;
}
/* Settings 2-col grid — ровная плитка полей; min-height у подписи выравнивает
   инпуты в строке, когда одна подпись в 1 строку, а соседняя переносится в 2. */
.settings-grid {
  display: grid;
  grid-template-columns: 1fr 1fr;
  gap: 10px 12px;
  align-items: start;
}
.settings-grid .field-label { min-height: 24px; }
/* «%»-суффикс в поле: вместо «, %» в подписи — единица справа внутри инпута */
.fld-pct { position: relative; display: block; }
.fld-pct > .input { padding-right: 22px; }
.fld-pct::after {
  content: "%";
  position: absolute;
  right: 9px; top: 50%;
  transform: translateY(-50%);
  font-size: 11px;
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  pointer-events: none;
}

/* ── Настройки группы (левая колонка): чёрная подложка панелей, чтобы поля/чипы «всплывали» (как в макете) ── */
.group-settings-panel .block {
  background: var(--bg-app);
  border-color: var(--border-strong);
}
/* ровная плитка полей в 2-колоночной сетке — без линий-разделителей и лишних отступов под ячейками */
.group-settings-panel .settings-grid .settings-field {
  padding-top: 0;
  padding-bottom: 0;
  border-bottom: none;
}
/* неактивные фильтр-чипы — на тон светлее подложки (как .mchip в макете), чтобы не сливались с чёрным */
.group-settings-panel .rating-chip:not([data-active="true"]) {
  background: var(--bg-neutral);
  border-color: var(--border-strong);
}
/* chips чуть крупнее — только внутри настроек группы */
.group-settings-panel .rating-chip { font-size: 12px; padding: 4px 10px; }
/* правая колонка «Управление списком»: акцентные элементы на чёрной подложке (как в макете) */
.group-settings-panel .tab.active .count {
  background: var(--bg-brand);
  color: var(--text-on-brand);
}
.group-settings-panel .tab:not(.active) {
  color: var(--text-tertiary);
}

/* Глобальные правила: чёрная подложка модалки (как в макете), серые блоки «всплывают» на ней */
.modal.rules-modal { background: var(--bg-app); border-color: var(--border-strong); }
.rules-modal > .block-head { background: var(--bg-container); }

/* ============================================================
   Глобальные правила — каркас модалки (эталон handoff/global-rules/desktop.dc.html):
   шапка (.gr-head) + прокручиваемое тело (.gr-body) + липкий футер (.gr-foot).
   Одна колонка, блоки стопкой. Мобильная версия (≤640) — ниже в @media: полноэкранный
   лист + карточки вместо таблиц.
   ============================================================ */
.modal.gr-modal {
  width: 880px; max-width: calc(100vw - 32px);
  display: flex; flex-direction: column;
  max-height: calc(100vh - 56px);
  overflow: hidden; padding: 0;
}
.gr-modal .gr-head {
  flex: 0 0 auto;
  display: flex; align-items: center; gap: 12px;
  padding: 14px 18px;
  border-bottom: 1px solid var(--border-default);
  background: var(--bg-container);
}
.gr-modal .gr-body {
  flex: 1 1 auto; min-height: 0;
  overflow-y: auto; overflow-x: hidden;
  padding: 16px 18px;
  display: flex; flex-direction: column; gap: 14px;
}
.gr-modal .gr-foot {
  flex: 0 0 auto;
  display: flex; align-items: center; gap: 10px;
  padding: 12px 18px;
  border-top: 1px solid var(--border-default);
  background: var(--bg-container);
}
.gr-modal .gr-foot-hint { font-size: 11px; margin-right: auto; }
.gr-modal .gr-scope-tag { flex: 0 0 auto; }

/* Кредит — два поля в ряд (мобайл: стопкой, см. @media) */
.gr-modal .gr-credit-row > .field { flex: 1 1 0; min-width: 0; }

/* Резерв — строка таблицы инструментов: Инструмент | Вес | Вкл | Удалить */
.gr-modal .gr-res-list { display: flex; flex-direction: column; }
.gr-modal .gr-res-row {
  display: grid;
  grid-template-columns: 1.5fr 88px 44px 34px;
  gap: 10px; align-items: center;
  padding: 9px 0;
  border-bottom: 1px solid var(--border-default);
}
.gr-modal .gr-res-row:last-child { border-bottom: 0; }
.gr-modal .gr-res-head { padding-top: 0; border-bottom: 1px solid var(--border-strong); }
.gr-modal .gr-res-c-sw { display: flex; justify-content: center; }
.gr-modal .gr-res-weight-cell { display: block; min-width: 0; }
.gr-modal .gr-res-weight-label { display: none; }      /* подпись «Вес, %» только в мобильной карточке */
.gr-modal .gr-res-weight { width: 100%; padding: 7px 10px; }
.gr-modal .gr-ellipsis { overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }

/* ЧС — пустое состояние и карточный список (карточки видны только на мобайле) */
.gr-modal .gr-empty {
  padding: 16px 14px; text-align: center; color: var(--text-tertiary);
  font-size: 12px; border: 1px dashed var(--border-default); border-radius: 10px;
}
.gr-modal .gr-bl-cards { display: none; flex-direction: column; gap: 8px; min-width: 0; }
.gr-modal .gr-bl-card {
  background: var(--bg-app); border: 1px solid var(--border-default);
  border-radius: 10px; padding: 10px 12px;
}
.gr-modal .gr-bl-card-head { display: flex; align-items: center; justify-content: space-between; gap: 10px; }
.gr-modal .gr-bl-card-head .btn { flex: 0 0 auto; }
.gr-modal .gr-bl-card-title { white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.gr-modal .gr-bl-card-sub { font-size: 10.5px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; display: block; margin-top: 3px; }

/* ============================================================
   Глобальные правила · МОБАЙЛ (≤640): оба модала (правила + редактор ЧС) — полноэкранные
   листы без горизонт-вылета. Корень разъезда был в инлайне width:880/680 на .modal —
   перебиваем !important и растягиваем на весь вьюпорт. Резерв и ЧС — карточками.
   ============================================================ */
@media (max-width: 640px) {
  /* .scrim хостит лист во всю площадь (specificity :has(...) > generic .scrim) */
  .scrim:has(> .gr-modal) { place-items: stretch; padding: 0; }
  /* .modal.gr-modal — (0,2,0), чтобы перебить десктопный max-height:calc(100vh-56px) выше.
     position:fixed + inset:0 жёстко прибивает лист к ВЬЮПОРТУ (а не к flow внутри .scrim),
     чтобы низ листа = низ экрана и футер .gr-foot («Сохранить») всегда был на экране.
     Раньше лист был position:static-потомком грид-скрима и стартовал с offset родителя
     (.ovr-screen со своим transform/padding) → height:100dvh свешивался ниже фолда и срезал
     «Сохранить». NB: чтобы fixed реально считался от вьюпорта, у предка-экрана убран transform
     входной анимации — см. .ovr-screen.fadein → fadein-noshift ниже (ПУНКТ про transform). */
  .modal.gr-modal {
    position: fixed; inset: 0;
    width: 100% !important; max-width: 100% !important;
    height: 100dvh; max-height: 100dvh;
    margin: 0; min-width: 0; border: 0; border-radius: 0;
  }
  .gr-modal .gr-head { padding: 12px 14px; }
  .gr-modal .gr-body { padding: 14px; gap: 12px; }
  .gr-modal .gr-foot { padding: 11px 14px 14px; box-shadow: 0 -8px 20px rgba(0,0,0,.3); }
  /* в шапке прячем тег «действует на весь счёт» и подпись-хинт в футере — узко */
  .gr-modal .gr-scope-tag, .gr-modal .gr-foot-hint { display: none; }
  /* футер: две кнопки во всю ширину (как в эталоне mobile.dc.html) */
  .gr-modal .gr-foot .btn { flex: 1 1 0; justify-content: center; }

  /* Кредит — поля стопкой. RULES-1: на мобайле перебиваем десктопный
     flex:1 1 0 у .field (иначе flex-basis:0 в колонке схлопывает поля по высоте —
     подпись и число налезали друг на друга). Каждое поле — натуральная высота,
     во всю ширину; инпут тоже 100% и min-width:0 (без горизонт-вылета). */
  .gr-modal .gr-credit-row { flex-direction: column; gap: 10px; align-items: stretch; }
  .gr-modal .gr-credit-row > .field { flex: 0 0 auto; width: 100%; min-width: 0; }
  .gr-modal .gr-credit-row > .field > .input { width: 100%; min-width: 0; }

  /* Резерв — каждая строка превращается в карточку: верх (инфо | вкл+удалить),
     низ — поле «Вес, %» во всю ширину. Шапку столбцов прячем. */
  .gr-modal .gr-res-head { display: none; }
  .gr-modal .gr-res-row {
    display: grid;
    grid-template-columns: minmax(0, 1fr) auto auto;
    grid-template-areas: "info sw del" "weight weight weight";
    gap: 8px; align-items: center;
    background: var(--bg-app); border: 1px solid var(--border-default);
    border-radius: 10px; padding: 10px 11px; margin-bottom: 8px;
  }
  .gr-modal .gr-res-row:last-of-type { margin-bottom: 0; }
  .gr-modal .gr-res-c-info { grid-area: info; }
  .gr-modal .gr-res-c-sw { grid-area: sw; }
  .gr-modal .gr-res-c-del { grid-area: del; }
  .gr-modal .gr-res-weight-cell {
    grid-area: weight; display: flex; flex-direction: column; gap: 5px;
  }
  .gr-modal .gr-res-weight-label { display: block; }
  .gr-modal .gr-res-weight { text-align: left !important; }

  /* ЧС — десктоп-таблицы прячем, показываем карточки */
  .gr-modal .gr-bl-desk { display: none; }
  .gr-modal .gr-bl-cards { display: flex; }
}

/* Switch */
.switch {
  width: 32px; height: 18px;
  background: var(--bg-neutral-press);
  border-radius: 999px;
  position: relative;
  display: inline-block;
  cursor: pointer;
  border: 0;
  padding: 0;
  flex-shrink: 0;
  transition: background .15s;
}
.switch::after {
  content: ""; position: absolute; left: 2px; top: 2px;
  width: 14px; height: 14px; border-radius: 50%;
  background: var(--text-primary);
  transition: transform .15s, background .15s;
}
.switch.on { background: var(--bg-brand); }
.switch.on::after { transform: translateX(14px); background: var(--text-on-brand); }

/* Segmented control */
.seg { display: inline-flex; background: var(--bg-neutral); border: 1px solid var(--border-default); border-radius: 4px; padding: 2px; }
.seg button {
  border: 0; background: transparent; color: var(--text-secondary);
  padding: 4px 10px; font-size: 11px; border-radius: 3px; cursor: pointer; font-family: inherit; font-weight: 500;
}
.seg button.active { background: var(--bg-container); color: var(--text-primary); box-shadow: 0 1px 2px oklch(0 0 0 / 0.35); }

/* Group type picker in the create-group modal */
.strategy-type-select {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: 8px;
  width: 100%;
  padding: 4px;
  background: var(--bg-neutral);
  border: 1px solid var(--border-default);
  border-radius: 8px;
}
.strategy-type-select button {
  position: relative;
  min-height: 42px;
  border: 1px solid transparent;
  border-radius: 6px;
  background: transparent;
  color: var(--text-secondary);
  font-family: inherit;
  font-size: 13px;
  font-weight: 700;
  cursor: pointer;
  transition: background .15s, border-color .15s, color .15s, box-shadow .15s;
}
.strategy-type-select button:hover {
  background: var(--bg-neutral-hover);
  color: var(--text-primary);
}
.strategy-type-select button.active {
  background: var(--bg-brand-tint);
  border-color: var(--border-brand);
  color: var(--text-primary);
  box-shadow: inset 0 0 0 1px oklch(from var(--color-brand-400) l c h / 0.28);
}
.strategy-type-select button.active::before {
  content: "";
  display: inline-block;
  width: 7px;
  height: 7px;
  margin-right: 8px;
  border-radius: 50%;
  background: var(--color-brand-400);
  box-shadow: 0 0 0 3px oklch(from var(--color-brand-400) l c h / 0.18);
  vertical-align: 1px;
}

/* Mini tester table inside the bond settings drawer */
.mini-tester-table {
  border: 1px solid var(--border-default);
  border-radius: 6px;
  overflow-x: auto;
  overflow-y: hidden;
  background: var(--bg-container);
  min-height: 348px;
}
.mini-tester-table::-webkit-scrollbar { height: 8px; }
.mini-tester-table::-webkit-scrollbar-thumb { background: var(--color-neutral-800); border-radius: 999px; }
.mini-tester-head,
.mini-tester-row {
  display: grid;
  grid-template-columns: 28px 76px 90px 76px 76px 58px 92px 50px 76px 68px 78px 82px 70px 44px 70px 88px;
  align-items: center;
  min-width: 1090px;
}
.mini-tester-head {
  min-height: 28px;
  background: var(--bg-neutral);
  border-bottom: 1px solid var(--border-default);
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  font-size: 9px;
  letter-spacing: .04em;
  text-transform: uppercase;
}
.mini-tester-row {
  min-height: 32px;
  border-bottom: 1px solid var(--border-default);
  font-family: var(--font-mono);
  font-size: 11px;
}
.mini-tester-row:last-child { border-bottom: 0; }
.mini-tester-row.recommended { background: var(--bg-success-tint); }
/* «Тестеры»: лучшая строка прогона (первая строка стр.1, отсортировано по чистому доходу) — лёгкий бренд-тон. */
.mini-tester-row.best { background: var(--bg-brand-tint); }
table.t.tester-results-table tbody tr.best { background: var(--bg-brand-tint); }
.mini-tester-row.applied {
  box-shadow: inset 2px 0 0 color-mix(in srgb, var(--border-success) 72%, transparent);
}
.mini-tester-row.applied-flash {
  animation: testerApplyFlash 1.25s ease-out;
}
.mini-tester-head > span,
.mini-tester-row > span {
  padding: 6px 8px;
  white-space: nowrap;
}
.mini-tester-row > button {
  justify-self: end;
  margin-right: 6px;
}
.mini-tester-pagination {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 8px;
  padding: 6px 0 2px;
}
/* ===== Тестер бумаги · мобильные карточки предложений (эталон mobile.dc.html стр.210-211).
   Десктоп (≥641) НЕ трогаем: видна .mini-tester-table, .mt-cards скрыты. ===== */
.mt-cards { display: none; }
@media (max-width: 640px) {
  /* широкая 16-колоночная таблица распирала оверлей → прячем, показываем карточки */
  .mini-tester-table.mt-desk { display: none; }
  .mt-cards { display: flex; flex-direction: column; gap: 8px; min-width: 0; }
  .mt-card { padding: 10px 12px; }
  .mt-card-head { display: flex; align-items: center; justify-content: space-between; gap: 8px; }
  .mt-card-title { font-weight: 700; font-size: 13px; }
  .mt-card-title .pos { color: var(--text-success); }
  .mt-card-title .neg { color: var(--text-error); }
  /* мини-KPI 5 в ряд — как .cgrid repeat(5,1fr) в эталоне */
  .mt-card .cgrid.mt-card-grid { display: grid; grid-template-columns: repeat(5, 1fr); gap: 7px 8px; margin-top: 8px; }
  .mt-card .cgrid.mt-card-grid .kpi { display: flex; flex-direction: column; gap: 1px; min-width: 0; }
  .mt-card .cgrid.mt-card-grid .kpi .k {
    font-size: 8.5px; font-family: var(--font-mono); text-transform: uppercase;
    letter-spacing: .03em; color: var(--text-tertiary); white-space: nowrap;
  }
  .mt-card .cgrid.mt-card-grid .kpi .v { font-size: 12px; font-weight: 600; font-family: var(--font-mono); white-space: nowrap; }
  /* второстепенные показатели — переносимой строкой, ничего не теряем (keep.md) */
  .mt-card-meta {
    display: flex; flex-wrap: wrap; gap: 4px 12px; margin-top: 9px; padding-top: 8px;
    border-top: 1px solid var(--border-default); font-size: 10.5px; color: var(--text-tertiary);
  }
  .mt-card-meta b { color: var(--text-secondary); font-weight: 600; margin-left: 3px; }
  .mt-card-meta b.pos { color: var(--text-success); }
  .mt-card-meta b.neg { color: var(--text-error); }
  .mt-card.best { border-color: var(--border-brand); }
  .mt-card.recommended { background: var(--bg-success-tint); }
  .mt-card.applied { box-shadow: inset 3px 0 0 var(--border-success); }
  .mt-card.applied-flash { animation: testerApplyFlash 1.25s ease-out; }
  .mt-cards-empty {
    padding: 14px 12px; text-align: center; color: var(--text-tertiary);
    font-size: 12px; border: 1px dashed var(--border-default); border-radius: 10px;
  }
}
.mini-run-progress {
  display: grid;
  gap: 5px;
  padding: 8px 9px;
  border: 1px solid var(--border-default);
  border-radius: 6px;
  background: var(--bg-container);
}
.mini-run-progress-top {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 10px;
  color: var(--text-secondary);
  font-size: 11px;
}
.mini-run-progress-top b {
  color: var(--text-primary);
}
.mini-run-progress-bar {
  height: 7px;
  overflow: hidden;
  border-radius: 999px;
  background: var(--color-neutral-900);
  border: 1px solid var(--border-default);
}
.mini-run-progress-bar i {
  display: block;
  height: 100%;
  min-width: 4px;
  border-radius: inherit;
  background: linear-gradient(90deg, var(--bg-brand), var(--color-success-500));
  transition: width .4s ease;
}
.mini-run-progress.running .mini-run-progress-bar i {
  background-size: 180% 100%;
  animation: portfolioProgressFlow 1.15s linear infinite;
}
.mini-run-progress.done .mini-run-progress-bar i {
  background: var(--color-success-500);
}
.mini-run-progress-detail {
  min-height: 13px;
  color: var(--text-tertiary);
  font-size: 10px;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}

/* Full tester: one viewport, chart + dense results table */
.tester-screen .block-head {
  min-height: 44px;
  padding: 10px 16px;
}
.tester-top-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) 280px;
  gap: 10px;
  align-items: stretch;
}
.tester-results-wrap {
  overflow: auto;
}
table.t.tester-results-table {
  table-layout: auto;
  font-size: 11px;
}
table.t.tester-results-table thead th {
  padding: 8px 8px;
  font-size: 9px;
  letter-spacing: .02em;
}
table.t.tester-results-table tbody td {
  padding: 9px 8px;
}
/* Выбранная строка (рисуется на графике) — info-акцент (inset-полоса + лёгкий info-тон),
   как в эталоне testers/desktop.dc.html (tr.active-row). Перебивает бренд-тон .best, чтобы
   выбор лучшей строки читался отдельно от «лучшего варианта». */
table.t.tester-results-table tbody tr.active {
  background: var(--bg-info-tint);
  box-shadow: inset 3px 0 0 0 var(--color-info-500);
}
/* Чистый доход — главная цифра, всегда bold (эталон font-weight:700). */
table.t.tester-results-table tbody td.tester-net { font-weight: 700; }
/* Цвет по знаку для Чистого дохода и Итого с НКД. Дженерик `table.t tbody td`
   красит td в --text-secondary и перебивает классы .pos/.neg (0,1,1 > 0,1,0),
   поэтому повышаем специфичность scoped-правилом (эталон: суммы цвет по знаку). */
table.t.tester-results-table tbody td.pos { color: var(--text-success); }
table.t.tester-results-table tbody td.neg { color: var(--text-error); }
/* Откр. сумма — приглушённая (эталон: muted). Тот же приоритет-фикс, что и для знака. */
table.t.tester-results-table tbody td.muted { color: var(--text-tertiary); }
/* Тот же приоритет-фикс для «Позиций» (PnL/НКД) и «Истории» (Прибыль/Купон/Чистая):
   дженерик `table.t tbody td` (--text-secondary, спец 0,1,1) перебивал .pos/.neg/.muted —
   повышаем специфичность scoped-правилами (эталон: знаковые суммы цвет по знаку). */
.pos-screen table.t tbody td.pos,
.hist-screen table.t tbody td.pos { color: var(--text-success); }
.pos-screen table.t tbody td.neg,
.hist-screen table.t tbody td.neg { color: var(--text-error); }
.pos-screen table.t tbody td.muted,
.hist-screen table.t tbody td.muted { color: var(--text-tertiary); }
/* «История»: строка купонного начисления — положительные суммы (Купон/Чистая) в инфо-тоне
   (эталон: color:var(--text-info)); перекрывает .pos на этой строке. */
.hist-screen table.t tbody tr.hist-coupon-row td.pos { color: var(--text-info); }
table.t.tester-results-table tbody tr.applied {
  box-shadow: inset 2px 0 0 color-mix(in srgb, var(--border-success) 72%, transparent);
}
table.t.tester-results-table tbody tr.applied-flash {
  animation: testerApplyFlash 1.25s ease-out;
}
table.t.tester-results-table .btn.sm {
  padding: 4px 7px;
  font-size: 10px;
}
table.t.tester-results-table th,
table.t.tester-results-table td {
  white-space: nowrap;
}
.tester-pagination {
  display: flex;
  align-items: center;
  justify-content: center;
  flex-wrap: wrap;
  gap: 6px;
  padding: 12px;
  border-top: 1px solid var(--border-default);
}
.tester-period-label {
  display: inline-flex;
  align-items: center;
  width: fit-content;
  padding: 3px 7px;
  border: 1px solid var(--border-default);
  border-radius: 4px;
  background: var(--bg-container-2);
  color: var(--text-secondary);
  font-family: var(--font-mono);
  font-size: 11px;
}
.bond-full-tester-modal {
  width: min(1560px, 96vw);
  max-height: 94vh;
  overflow: hidden;
  display: flex;
  flex-direction: column;
}
.bond-full-tester-body {
  min-height: 0;
  overflow: auto;
}
.bond-full-tester-body .tester-screen {
  min-height: 0;
  padding: 12px;
}
/* Подсказка горизонт-скролла под таблицей результатов — только мобайл. */
.tester-scroll-hint { display: none; }

/* ===== Полный тестер · мобайл (≤640px), эталон testers/mobile.dc.html.
   Десктоп (≥641) НЕ трогаем. Таблица 16 колонок 1:1 — ничего не режем, даём
   горизонт-скролл внутри обёртки + подсказку. Страница вбок не вылезает. ===== */
@media (max-width: 640px) {
  /* min-width:0 по цепочке: иначе широкая таблица распирает .scroll и вся страница едет вбок */
  .tester-screen { padding: 14px; gap: 12px; min-width: 0; }
  .tester-screen > .block, .tester-screen .block-body { min-width: 0; max-width: 100%; }

  /* Шапка: заголовок и кнопка в столбик, кнопка — во всю ширину (мин. таргет 44px) */
  .tester-screen > .row { flex-wrap: wrap; align-items: stretch; gap: 10px; }
  .tester-screen > .row > .col { min-width: 0; flex: 1 1 100%; }
  .tester-screen > .row > .row { flex: 1 1 100%; }
  .tester-screen > .row > .row .btn.primary { flex: 1; justify-content: center; min-height: 44px; }
  .tester-screen .h1 { font-size: 18px; }

  /* «Предложения»: шапка блока переносит селекты под заголовок, селекты тянутся 1:1 */
  .tester-screen .block-head { flex-wrap: wrap; row-gap: 8px; }
  .tester-screen .block-head > .row.gap-6 { flex: 1 1 100%; }
  .tester-screen .block-head > .row.gap-6 > .select { flex: 1; min-width: 0; }

  /* Все 16 колонок держат размер → скролл вбок ВНУТРИ обёртки (как mobile.dc.html min-width:920) */
  .tester-results-wrap { overflow-x: auto; -webkit-overflow-scrolling: touch; max-width: 100%; }
  table.t.tester-results-table { min-width: 920px; }

  .tester-scroll-hint { display: block; font-size: 10px; padding: 8px 2px 0; }
}

/* Admin */
.admin-catalog {
  position: relative; /* контекст позиционирования: ни один потомок не «всплывает» поверх соседних блоков */
  min-height: 0;
  overflow: hidden;
}
.admin-table-wrap {
  max-height: calc(100vh - 330px);
  min-height: 360px;
  overflow: auto;
  border-top: 1px solid var(--border-default);
}
.admin-table {
  min-width: 1180px;
}
.admin-table tbody td {
  vertical-align: middle;
}
.admin-table .sec {
  max-width: 360px;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.admin-payload {
  margin: 0;
  padding: 14px 18px;
  max-height: 320px;
  overflow: auto;
  background: #070b0f;
  border-top: 1px solid var(--border-default);
  color: var(--text-secondary);
  font: 12px/1.55 var(--font-mono);
  white-space: pre-wrap;
}
.admin-incidents-table {
  min-width: 1260px;
}
.admin-cpu-cell {
  display: grid;
  grid-template-columns: 72px minmax(90px, 1fr);
  align-items: center;
  gap: 10px;
}
.admin-cpu-bar {
  display: block;
  height: 6px;
  overflow: hidden;
  border-radius: 999px;
  background: rgba(255, 255, 255, 0.08);
}
.admin-cpu-bar i {
  display: block;
  height: 100%;
  min-width: 2px;
  border-radius: inherit;
  background: var(--bg-brand);
}
.admin-process-warm {
  background: rgba(255, 196, 74, 0.06);
}
.admin-process-hot {
  background: rgba(255, 92, 92, 0.09);
}
.admin-process-hot .admin-cpu-bar i {
  background: #ff6b6b;
}

/* ===== АДМИНКА · МОБИЛЬНАЯ АДАПТАЦИЯ (≤640) =====================================
   Десктоп (≥641) НЕ трогаем: .admin-cards скрыты, видны таблицы (.admin-desk).
   Эталон — handoff/admin/mobile.dc.html. На мобайле широкие таблицы → карточки
   (Счета/Инструменты/Процессы), сегмент-контрол сверху из 3 чипов. */
.admin-cards { display: none; }

/* Десктоп-паддинг корня админки (раньше был inline style padding:24 на .admin-screen —
   inline перебивал мобильный padding:14px и мешал @media-правилам; вынесли в базовый класс,
   чтобы мобильные overrides ниже работали по специфичности). Десктоп без изменений. */
.admin-screen { padding: 24px; }

@media (max-width: 640px) {
  /* min-width:0 по цепочке: иначе остаточные широкие таблицы распирают страницу вбок.
     min-height:auto обязателен: корень имеет inline min-height:100%, который в цепочке
     .scroll(flex:1;min-height:0) клампит экран по высоте вьюпорта — контент ниже фолда
     вылезает за бокс, НЕ удлиняет scrollHeight, и страница НЕ скроллится вертикально
     (последние карточки/кнопки прячутся под .mobile-tabbar). Распускаем как у
     .bonds-screen-root — экран течёт по высоте контента, .scroll скроллит, padding-bottom
     (74px) даёт зазор под таб-баром. Десктоп (≥641) не затрагивается. */
  .admin-screen { padding: 14px; gap: 12px; min-width: 0; min-height: auto !important; }
  .admin-screen > .block, .admin-screen .block-body, .admin-screen .admin-catalog { min-width: 0; max-width: 100%; }
  .admin-screen .h1 { font-size: 18px; }

  /* Шапка: заголовок + сегмент-контрол в столбик; сегмент во всю ширину сверху (стр.45) */
  .admin-screen > .row { flex-wrap: wrap; align-items: stretch; gap: 10px; }
  .admin-screen > .row > .col { min-width: 0; flex: 1 1 100%; }
  .admin-screen > .row > .row.gap-8 { flex: 1 1 100%; min-width: 0; flex-wrap: wrap; row-gap: 8px; }

  /* Сегмент разделов (1-й .seg в шапке) = компактный сегмент-контрол как .seg-m в эталоне.
     На мобайле доступны все 5 разделов — Счета(1)·Инструменты(2)·Инциденты(3)·Процессы(4)·Стримы(5);
     сегмент скроллит вбок (как фильтр-сегменты ниже), кнопки своей ширины, не ужимаются. */
  .admin-screen > .row > .row.gap-8 > .seg {
    flex: 1 1 100%; display: flex; min-width: 0;
    background: var(--bg-neutral); border: 1px solid var(--border-default);
    border-radius: 10px; padding: 3px; gap: 0;
    overflow-x: auto; -webkit-overflow-scrolling: touch; scrollbar-width: none;
  }
  .admin-screen > .row > .row.gap-8 > .seg::-webkit-scrollbar { display: none; }
  .admin-screen > .row > .row.gap-8 > .seg button {
    flex: 0 0 auto; white-space: nowrap; padding: 8px 4px; font-size: 12px; font-weight: 600;
    border: 0; border-radius: 7px;
  }
  /* кнопки прогрузки/синка инструментов (в шапке рядом с сегментом) — в столбик, во всю ширину */
  .admin-screen > .row > .row.gap-8 > .btn { flex: 1 1 100%; justify-content: center; min-height: 42px; }

  /* KPI — 3 в ряд (эталон .kpi2 flex:1), плотнее по вертикали. Перебиваем общий 2-кол на ≤640. */
  .admin-screen .kpi-grid { grid-template-columns: repeat(3, minmax(0, 1fr)); }
  .admin-screen .kpi-grid .kpi-cell { padding: 9px 10px; gap: 2px; }
  .admin-screen .kpi-grid .kpi-label { font-size: 9px; letter-spacing: .02em; }
  .admin-screen .kpi-grid .kpi-value { font-size: 19px; line-height: 1.05; }
  .admin-screen .kpi-grid .kpi-sub { font-size: 9px; line-height: 1.15; }

  /* Блок-шапка каталога: заголовок + тулбар (поиск/сегменты) в столбик */
  .admin-screen .admin-catalog .block-head { flex-wrap: wrap; row-gap: 10px; padding: 12px 14px; }
  .admin-screen .admin-catalog .block-head > .col { flex: 1 1 100%; min-width: 0; }
  .admin-screen .admin-catalog .block-head > .row.gap-8 { flex: 1 1 100%; min-width: 0; flex-wrap: wrap; row-gap: 8px; }
  /* поиск — во всю ширину (стр.52/90) */
  .admin-screen .admin-catalog .block-head .input { flex: 1 1 100%; width: auto !important; min-width: 0; min-height: 42px; }
  /* фильтр-сегменты (статус/класс/режим) — чипы скроллом, не толкают ширину (стр.51/89) */
  .admin-screen .admin-catalog .block-head .seg {
    flex: 1 1 100%; min-width: 0; max-width: 100%;
    overflow-x: auto; -webkit-overflow-scrolling: touch; scrollbar-width: none;
  }
  .admin-screen .admin-catalog .block-head .seg::-webkit-scrollbar { display: none; }
  .admin-screen .admin-catalog .block-head .seg button { flex: 0 0 auto; white-space: nowrap; }

  /* прячем десктоп-таблицы, показываем карточки */
  .admin-screen .admin-desk { display: none; }
  .admin-screen .admin-cards {
    display: flex; flex-direction: column; gap: 10px; min-width: 0;
    padding: 0 14px 16px;
  }
  .admin-screen .admin-cards-empty {
    padding: 16px 12px; text-align: center; color: var(--text-tertiary);
    font-size: 12px; border: 1px dashed var(--border-default); border-radius: 10px;
  }

  /* --- Карточка СЧЁТА (стр.55-80) --- */
  .admin-acc-card { padding: 12px; }
  .admin-acc-card-head { display: flex; align-items: center; justify-content: space-between; gap: 8px; }
  .admin-acc-card-id { font-weight: 700; font-size: 15px; }
  .admin-acc-card-title { font-size: 11px; }
  .admin-acc-card-meta { gap: 2px; margin-top: 6px; }
  .admin-acc-card-meta .ell { display: block; max-width: 100%; }
  .admin-acc-card-meta .muted { font-size: 11px; }
  .admin-acc-card-token { font-size: 10px; max-width: 200px; }
  .admin-acc-card-date { font-size: 10px; flex-shrink: 0; }
  /* блок «лицензия/сервер» = коробка как .lic-box (стр.58/66) */
  .admin-acc-card-lic, .admin-acc-lic-empty {
    background: var(--bg-neutral); border: 1px solid var(--border-default);
    border-radius: 10px; padding: 9px 10px; margin-top: 9px;
  }
  .admin-acc-lic-empty .muted { font-size: 11px; }
  /* причина отклонения — error-цвет (стр.78) */
  .admin-acc-card-reason { margin-top: 8px; font-size: 11px; color: var(--text-error); }
  /* действия — крупные кнопки ≥42px в ряд (стр.59/72/79) */
  .admin-acc-card-actions { margin-top: 10px; }
  .admin-actbtn { flex: 1; justify-content: center; min-height: 42px; }

  /* Переразложение LicenseAdminCell внутри карточки одобренного: компактные строки
     «метка · число · дн · кнопка», теги статуса в ряд сверху (стр.67-70).
     Состав контролов 1:1 с десктопом — режем только плотность/раскладку. */
  .admin-acc-card-lic .col { min-width: 0; gap: 8px !important; }
  .admin-acc-card-lic .col > .row { flex-wrap: wrap; gap: 6px; align-items: center; }
  .admin-acc-card-lic .btn.sm { min-height: 34px; }
  /* поля ввода дней — компактные, выровнены вправо, тянутся в строке */
  .admin-acc-card-lic .input.mono {
    width: auto !important; flex: 1 1 60px; min-width: 52px; min-height: 36px;
    padding: 6px 10px; text-align: right;
  }
  .admin-acc-card-lic .sec { font-size: 11px; }

  /* --- Карточка ИНСТРУМЕНТА (стр.91-92) --- */
  .admin-ins-card { padding: 12px; }
  .admin-ins-card-top { display: flex; align-items: center; justify-content: space-between; gap: 8px; }
  .admin-ins-card-tick { font-weight: 700; font-size: 14px; }
  .admin-ins-card-sub { font-size: 10px; display: block; max-width: 100%; }
  .admin-ins-card-foot { display: flex; align-items: center; justify-content: space-between; gap: 8px; margin-top: 8px; }
  .admin-ins-card-line { font-size: 11px; min-width: 0; }
  .admin-ins-card-state { flex-shrink: 0; }
  .admin-ins-card-status { margin-top: 6px; font-size: 10px; display: block; max-width: 100%; }

  /* --- Карточка ПРОЦЕССА (стр.100-101) --- */
  .admin-proc-card { padding: 12px; }
  .admin-proc-card.is-warm { background: rgba(255, 196, 74, 0.06); }
  .admin-proc-card.is-hot { background: rgba(255, 92, 92, 0.09); }
  .admin-proc-card-head { display: flex; align-items: center; justify-content: space-between; gap: 8px; }
  .admin-proc-card-name { font-weight: 600; font-size: 13px; min-width: 0; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
  .admin-proc-card-pid { font-size: 11px; flex-shrink: 0; }
  .admin-proc-card-foot { display: flex; align-items: center; justify-content: space-between; gap: 8px; margin-top: 7px; font-size: 11px; }
  .admin-proc-card-note { display: block; margin-top: 6px; font-size: 10px; max-width: 100%; }

  /* Безопасность для не-эталонных разделов (Инциденты/Стримы): если открыты на мобайле —
     таблица скроллит вбок внутри обёртки, страницу не распирает. */
  .admin-screen .admin-incidents-table,
  .admin-screen .admin-table { min-width: 0; }
  .admin-screen .admin-table-wrap:not(.admin-desk) { overflow-x: auto; -webkit-overflow-scrolling: touch; max-width: 100%; }
}

/* ============================================================
   Админка · мониторинг сервера (раздел «Процессы») — карточки-графики
   в стиле хостинг-панели: заголовок + значение-подпись + линейный график
   с осью времени снизу. Сетка 2×2 на десктопе, 1 колонка на мобайле.
   ============================================================ */
/* Высота области графика в одном месте — совпадает с props.height у MetricTimeSeriesChart
   (по умолчанию 150). Лочим её в CSS, чтобы строка грида измерялась по этой высоте,
   а не по «intrinsic» высоте <svg> (которая при width:100% + preserveAspectRatio="none"
   может посчитаться браузером иначе и схлопнуть строку → следующий блок наезжает сверху). */
.admin-metrics-block { --admin-metric-h: 150px; overflow: visible; }
.admin-metrics-grid {
  display: grid;
  grid-template-columns: 1fr 1fr;
  /* Явные строки auto: высота каждой строки = по контенту карточек (а не растяжка/схлоп).
     Две строки 2×2; если карточек больше — добавятся такие же auto-строки. */
  grid-auto-rows: auto;
  align-items: start;
  gap: 14px;
  padding: 16px;
  border-top: 1px solid var(--border-default);
}
.admin-metric-card {
  padding: 14px 16px 12px;
  display: flex;
  flex-direction: column;
  gap: 8px;
  min-width: 0;
}
.admin-metric-head {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 10px;
  flex-wrap: wrap;
}
.admin-metric-title { font-weight: 600; font-size: 13px; color: var(--text-primary); }
.admin-metric-value { font-family: var(--font-mono); font-size: 12px; font-weight: 700; }
/* Высота фиксирована (не только min-height): даёт строке грида детерминированную высоту,
   независимо от того, как браузер вычислит intrinsic-высоту <svg>. flex-shrink:0 —
   чтобы плот не сжимался в flex-колонке карточки. */
.admin-metric-plot { position: relative; width: 100%; height: var(--admin-metric-h, 150px); flex: 0 0 auto; }
/* svg тянется на всю высоту плота (а не полагается на свой height-атрибут): даже при
   глобальном сбросе svg{height:auto} высота останется = плоту, и график не «распухнет»
   по аспект-ратио, наезжая на соседние блоки. width:100% — на всю ширину. */
.admin-metric-plot svg { cursor: crosshair; display: block; width: 100%; height: 100%; }
.admin-metric-empty {
  display: flex; align-items: center; justify-content: center;
  height: var(--admin-metric-h, 150px); font-size: 12px; color: var(--text-tertiary);
}
/* Базовое (десктоп) состояние «загрузка/пусто» для панелей раздела «Процессы».
   На мобайле ниже переопределяется на карточку с пунктиром (.admin-screen scope). */
.admin-cards-empty {
  display: flex; align-items: center; justify-content: center;
  text-align: center; font-size: 12px; color: var(--text-tertiary);
}
.admin-metric-legend {
  display: flex; flex-wrap: wrap; gap: 4px 16px;
  padding-top: 2px;
}
.admin-metric-legend-item { display: flex; align-items: center; gap: 6px; font-size: 11px; }
.admin-metric-swatch { width: 16px; height: 3px; border-radius: 2px; flex-shrink: 0; }
.admin-metric-swatch-dash {
  height: 0; border-top: 2px dashed currentColor; background: none !important;
  border-radius: 0; opacity: 0.9;
}

@media (max-width: 640px) {
  .admin-metrics-grid { grid-template-columns: 1fr; gap: 12px; padding: 12px; }
  .admin-metric-card { padding: 12px 13px; }
}

/* Trading live area */
.trade-live-grid {
  flex: 0 0 auto;
  height: 325px;
  display: grid;
  /* Стакан 312px (макет ~249: grid-template-columns: 1fr 312px). Мобильный сплит — в @media ниже. */
  grid-template-columns: minmax(0, 1fr) 312px;
  border-bottom: 1px solid var(--border-default);
  background: var(--bg-container);
}

/* График, раскрывающийся ВНУТРИ строки выбранной бумаги (inline chart-on-row) */
.chart-inline-row > td { padding: 0; background: var(--bg-app); border-bottom: 1px solid var(--border-default); }
.chart-inline-reveal { overflow: hidden; animation: chart-inline-in .22s ease-out; }
@keyframes chart-inline-in {
  from { opacity: 0; transform: translateY(-8px); }
  to   { opacity: 1; transform: translateY(0); }
}
.chart-inline-stage .trade-live-grid { height: 300px; border-bottom: 0; }
.orderbook {
  border-left: 1px solid var(--border-default);
  display: flex;
  flex-direction: column;
  min-width: 0;
  min-height: 0;
  background: var(--bg-container);
  overflow: hidden;
}
.orderbook-head {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 8px;
  padding: 6px 10px;
  border-bottom: 1px solid var(--border-default);
}
.orderbook-grid {
  display: grid;
  grid-template-columns: 1fr 1fr 1fr;
  gap: 6px;
  align-items: center;
}
.orderbook-grid-head {
  padding: 4px 10px;
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  font-size: 9px;
  text-transform: uppercase;
  letter-spacing: .04em;
}
.orderbook-side {
  display: flex;
  flex-direction: column;
  gap: 0;
  padding: 0 8px;
}
.orderbook-row {
  position: relative;
  display: grid;
  grid-template-columns: 1fr 1fr 1fr;
  gap: 6px;
  align-items: center;
  min-height: 13px;
  padding: 0 2px;
  font-size: 9px;
  overflow: hidden;
}
.orderbook-row > span {
  position: relative;
  z-index: 1;
}
.orderbook-row i {
  position: absolute;
  top: 2px;
  bottom: 2px;
  opacity: .18;
}
.orderbook-row.ask i {
  right: 0;
  background: var(--color-error-500);
}
.orderbook-row.bid i {
  left: 0;
  background: var(--color-success-500);
}
.orderbook-row.mine {
  outline: 1px solid var(--color-warning-500);
  background: rgba(255, 196, 74, 0.12);
}
.orderbook-row b {
  position: absolute;
  right: 3px;
  z-index: 2;
  color: var(--text-warning);
  font-size: 8px;
  font-family: var(--font-mono);
  text-transform: uppercase;
  padding: 0 3px;
  border-radius: 3px;
  background: rgba(0, 0, 0, 0.35);
}
/* Пустая строка-плейсхолдер: держит высоту и симметрию стакана, без данных и depth-бара,
   приглушённая (еле заметная пунктирная черта по центру вместо цены/лота).
   Класс «ph», НЕ «empty» — глобальная утилита .empty навешивает padding:18px и ломает высоту. */
.orderbook-row.ph {
  pointer-events: none;
  opacity: .5;
}
.orderbook-row.ph::after {
  content: "";
  position: absolute;
  left: 8px;
  right: 8px;
  top: 50%;
  border-top: 1px dotted var(--border-default);
  opacity: .6;
}
.orderbook-mid {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 8px;
  /* не сжиматься во флекс-колонке стакана (иначе при полной глубине цену «съедает» до пары px) */
  flex: 0 0 auto;
  min-height: 0;
  padding: 1px 10px;
  margin: 2px 0;
  /* узкая полоса: только тонкие разделители сверху/снизу, без объёмной рамки/радиуса */
  border: 0;
  border-top: 1px solid var(--border-default);
  border-bottom: 1px solid var(--border-default);
  border-radius: 0;
  background: var(--bg-neutral);
  font-size: 11px;
  line-height: 1.35;
}
.orderbook-note {
  display: flex;
  align-items: center;
  gap: 6px;
  padding: 5px 10px;
  border-top: 1px solid var(--border-default);
  color: var(--text-warning);
  font-size: 10px;
  font-family: var(--font-mono);
}
.orderbook-note-list {
  display: flex;
  flex-wrap: wrap;
  gap: 6px;
}
.orderbook-note-list b {
  padding: 1px 5px;
  border: 1px solid var(--border-default);
  border-radius: 4px;
  color: var(--text-warning);
  font-weight: 700;
}
.orderbook-empty {
  flex: 1;
  display: grid;
  place-content: center;
  gap: 8px;
  padding: 20px;
  text-align: center;
  color: var(--text-tertiary);
  font-size: 12px;
}
.orderbook-empty svg {
  margin: 0 auto;
  color: var(--text-warning);
}
.orderbook-empty small {
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  font-size: 10px;
}

/* KPI Block grid */
.kpi-grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(180px, 1fr)); gap: 0; border: 1px solid var(--border-default); border-radius: 8px; overflow: hidden; background: var(--bg-container); }
.kpi-cell { padding: 16px; border-right: 1px solid var(--border-default); display: flex; flex-direction: column; gap: 6px; }
.kpi-cell:last-child { border-right: 0; }
.kpi-label { font-size: 10px; text-transform: uppercase; letter-spacing: .04em; color: var(--text-tertiary); font-family: var(--font-mono); }
.kpi-value { font-family: var(--font-mono); font-size: 22px; font-weight: 600; font-feature-settings: "tnum" 1, "lnum" 1; color: var(--text-primary); }
.kpi-sub   { font-size: 11px; color: var(--text-tertiary); font-family: var(--font-mono); display: flex; align-items: center; gap: 6px; }

/* Linkable row */
.row-link { color: var(--text-primary); cursor: pointer; }
.row-link:hover { color: var(--text-brand); }

/* Misc */
.divider-v { width: 1px; background: var(--border-default); align-self: stretch; }
.divider-h { height: 1px; background: var(--border-default); }
.spacer { flex: 1; }
.gap-4 { gap: 4px; }  .gap-6 { gap: 6px; }  .gap-8 { gap: 8px; } .gap-12 { gap: 12px; } .gap-16 { gap: 16px; } .gap-24 { gap: 24px; }
.col { display: flex; flex-direction: column; }
.row { display: flex; align-items: center; }
.row.wrap { flex-wrap: wrap; }
.fill { flex: 1; min-width: 0; }
.center { display: grid; place-items: center; }
.bar {
  height: 6px;
  background: var(--bg-neutral);
  border-radius: 999px;
  overflow: hidden;
  position: relative;
}
.bar > i { display: block; height: 100%; background: var(--bg-brand); border-radius: 999px; }
.bar > i.warn { background: var(--color-warning-500); }
.bar > i.error { background: var(--color-error-500); }
.bar > i.success { background: var(--color-success-500); }

/* Mini sparkline container */
.spark { display: block; }

/* Reason / cause callout */
.reason {
  display: flex;
  gap: 8px;
  padding: 8px 10px;
  background: var(--bg-neutral);
  border-left: 2px solid var(--color-neutral-600);
  border-radius: 2px;
  font-size: 11px;
  color: var(--text-secondary);
  line-height: 1.4;
}
.reason.warn  { background: var(--bg-warning-tint); border-color: var(--color-warning-500); color: var(--text-warning); }
.reason.error { background: var(--bg-error-tint);   border-color: var(--color-error-500);   color: var(--text-error); }
.reason.info  { background: var(--bg-info-tint);    border-color: var(--color-info-500);    color: var(--text-info); }
.reason.success { background: var(--bg-success-tint); border-color: var(--color-success-500); color: var(--text-success); }

/* Breadcrumb */
.crumb { display: flex; align-items: center; gap: 6px; font-family: var(--font-mono); font-size: 11px; color: var(--text-tertiary); }
.crumb a, .crumb span.link { color: var(--text-tertiary); cursor: pointer; }
.crumb a:hover, .crumb span.link:hover { color: var(--text-primary); }
.crumb .sep { opacity: .5; }
.crumb .current { color: var(--text-primary); }
.topbar-account { display: none; } /* виден только на мобилке (вместо хлебных крошек) */

/* Sidebar item active */
.nav-btn {
  display: flex;
  align-items: center;
  gap: 10px;
  padding: 7px 10px;
  background: transparent;
  border: 0;
  color: var(--text-secondary);
  font-size: 12px;
  font-weight: 500;
  border-radius: 4px;
  cursor: pointer;
  text-align: left;
  width: 100%;
  font-family: inherit;
  transition: background .15s, color .15s;
  white-space: nowrap;
}
.nav-btn > span:not(.count) { overflow: hidden; text-overflow: ellipsis; flex: 1; }
.nav-btn:hover { background: var(--bg-neutral); color: var(--text-primary); }
.nav-btn.active { background: var(--bg-brand-tint); color: var(--text-brand); }
.nav-btn .count { font-family: var(--font-mono); font-size: 10px; padding: 0 6px; border-radius: 999px; background: var(--bg-neutral); color: var(--text-tertiary); margin-left: auto; }
.nav-btn.active .count { background: var(--bg-brand); color: var(--text-on-brand); }
.nav-section-title { font-size: 9px; font-family: var(--font-mono); text-transform: uppercase; letter-spacing: .08em; color: var(--text-tertiary); padding: 12px 12px 6px; }

/* ============================================================
   AccountSwitcher — выпадающий список переключения счёта (NAV-1).
   Десктоп: триггер = карточка счёта в сайдбаре, меню — поповер под ней.
   Мобайл (≤640px): меню превращается в боттом-шит (см. медиазапрос ниже).
   ============================================================ */
.acct-switch { position: relative; }
.acct-switch-sidebar { margin: 10px; }

/* Триггер в сайдбаре — визуально как прежняя карточка счёта */
.acct-switch-trigger {
  width: 100%;
  display: flex; align-items: center; gap: 8px;
  padding: 10px 12px;
  background: var(--bg-container);
  border: 1px solid var(--border-default);
  border-radius: 6px;
  color: var(--text-primary);
  font-family: inherit;
  text-align: left;
  cursor: pointer;
  transition: border-color .15s, background .15s;
}
.acct-switch-trigger:hover { border-color: var(--border-brand); }
.acct-switch.open .acct-switch-trigger { border-color: var(--border-brand); }
.acct-switch-trigger-text { gap: 0; flex: 1; min-width: 0; }
.acct-switch-trigger-broker { font-size: 10px; color: var(--text-tertiary); }
.acct-switch-trigger-name {
  font-size: 12px; font-weight: 500;
  white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}

/* Триггер-чип в шапке (показан только на мобайле — десктопный .topbar-account скрыт CSS) */
.acct-switch-chip {
  display: inline-flex; align-items: center; gap: 5px;
  padding: 5px 8px;
  background: var(--bg-container);
  border: 1px solid var(--border-default);
  border-radius: 999px;
  color: var(--text-secondary);
  font-family: inherit;
  cursor: pointer;
}
.acct-switch-chip .mono { font-size: 12px; }
.acct-switch.open .acct-switch-chip { border-color: var(--border-brand); color: var(--text-brand); }

/* Слой-перехватчик клика вне меню (десктоп — прозрачный) */
.acct-switch-scrim { position: fixed; inset: 0; z-index: 200; }

/* Меню-поповер */
.acct-switch-menu {
  position: absolute;
  z-index: 201;
  top: calc(100% + 6px);
  left: 0; right: 0;
  display: flex; flex-direction: column;
  padding: 8px;
  border: 1px solid var(--border-default);
  border-radius: 10px;
  background: var(--bg-elevated);
  box-shadow: 0 18px 42px rgba(0, 0, 0, .4);
}
.acct-switch-topbar .acct-switch-menu {
  /* В шапке чип уже у правого края — раскрываем влево, фиксируем ширину. */
  left: auto; right: 0; min-width: 240px;
}
.acct-switch-menu-handle { display: none; }
.acct-switch-menu-head { padding: 2px 6px 8px; }
.acct-switch-menu-list { display: flex; flex-direction: column; gap: 2px; max-height: 52vh; overflow-y: auto; }

.acct-switch-item {
  display: flex; align-items: center; gap: 10px;
  width: 100%;
  padding: 8px 8px;
  border: 0; border-radius: 7px;
  background: transparent;
  color: var(--text-primary);
  font-family: inherit;
  text-align: left;
  cursor: pointer;
  transition: background .12s;
}
.acct-switch-item:hover { background: var(--bg-neutral-hover); }
.acct-switch-item.active { background: var(--bg-brand-tint); }
.acct-switch-item .dot { margin-left: 2px; }
.acct-switch-item-text { gap: 1px; flex: 1; min-width: 0; }
.acct-switch-item-top { gap: 6px; align-items: center; }
.acct-switch-item-id { font-size: 12px; font-weight: 600; }
.acct-switch-item-broker { font-size: 9px; padding: 0 6px; }
.acct-switch-item-sub { font-size: 10px; }
.acct-switch-item.active > svg { color: var(--text-brand); flex-shrink: 0; }

.acct-switch-manage {
  display: flex; align-items: center; gap: 8px;
  width: 100%;
  margin-top: 6px;
  padding: 9px 8px;
  border: 0; border-top: 1px solid var(--border-default);
  border-radius: 0 0 4px 4px;
  background: transparent;
  color: var(--text-secondary);
  font-family: inherit; font-size: 12px; font-weight: 500;
  text-align: left;
  cursor: pointer;
}
.acct-switch-manage:hover { color: var(--text-brand); }
.acct-switch-manage > span { flex: 1; }

/* ===== [NEW] Сворачивание сайдбара в rail + тултипы (ТОЛЬКО десктоп ≥641px).
   На мобайле .shell > aside{display:none} (см. @media ≤640) — rail там не рисуется,
   но тултип/правила всё равно гейтим в min-width:641, чтобы мобайл не затрагивать. */
@media (min-width: 641px) {
  /* Ширина сайдбара + плавный переход — в CSS, не инлайном: инлайновый width React
     при ре-рендере клампил под flex min-content и rail не сжимался до 60. Здесь width
     задаёт стабильный класс .rail (а minWidth:0 на aside снимает flex min-content).
     transition:width .15s — как в эталоне (desktop.dc.html). Прим.: в dev-превью CSS-
     переходы не тикают (headless), поэтому ширину там видно сразу в конечном значении;
     в реальном браузере переход анимируется. */
  .shell > aside { width: 220px; transition: width .15s; }
  .shell > aside.rail { width: 60px; }
  /* Свёрнутый сайдбар: иконки по центру, без перетекания горизонт-паддингов внутрь */
  .rail .nav-btn { padding-left: 0; padding-right: 0; }
  /* Подпись-тултип справа от иконки; по умолчанию скрыта, видна при наведении в rail */
  .nav-btn { position: relative; }
  .nav-btn .tip {
    display: none;
    position: absolute; left: 54px; top: 50%; transform: translateY(-50%);
    background: var(--bg-elevated); border: 1px solid var(--border-strong);
    color: var(--text-primary); font-size: 12px; padding: 5px 9px; border-radius: 6px;
    white-space: nowrap; z-index: 60; box-shadow: 0 6px 18px rgba(0,0,0,.4);
    pointer-events: none;
  }
  .rail .nav-btn:hover .tip { display: block; }
}

/* Chart container */
.chart-wrap { position: relative; background: var(--bg-container); }
.chart-axis text { fill: var(--text-tertiary); font-family: var(--font-mono); font-size: 10px; }
.chart-grid line { stroke: var(--border-default); stroke-dasharray: 2 4; }
.chart-channel-top, .chart-channel-bot { stroke: var(--color-info-400); stroke-width: 1; stroke-dasharray: 4 3; fill: none; }
.chart-channel-fill { fill: oklch(from var(--color-info-500) l c h / 0.04); }
/* touch-action: none — график «ловит» жест: ОДИН палец = полный пан (обе оси),
   ДВА пальца = пинч-зум + скролл страницы (всё в chart.jsx). */
.interactive-chart { cursor: grab; touch-action: none; user-select: none; }
.interactive-chart:active { cursor: grabbing; }
.interactive-chart.ruler-mode { cursor: crosshair; }
.chart-price-scale-hit { cursor: ns-resize; pointer-events: all; }
.chart-time-scale-hit { cursor: ew-resize; pointer-events: all; }
.chart-floating-tools {
  position: absolute;
  left: 10px;
  top: 8px;
  z-index: 3;
  display: flex;
  align-items: center;
  gap: 6px;
  pointer-events: auto;
}
.chart-tool {
  height: 24px;
  min-width: 28px;
  border: 1px solid var(--border-default);
  background: oklch(from var(--bg-container-2) l c h / 0.9);
  color: var(--text-secondary);
  border-radius: 4px;
  font: 700 13px/1 var(--font-mono);
  cursor: pointer;
}
.chart-tool:hover,
.chart-tool.active {
  border-color: var(--color-info-500);
  color: var(--text-info);
  background: var(--bg-info-tint);
}
.chart-source {
  height: 24px;
  display: inline-flex;
  align-items: center;
  padding: 0 8px;
  border: 1px solid var(--border-default);
  background: oklch(from var(--bg-container-2) l c h / 0.82);
  color: var(--text-tertiary);
  border-radius: 4px;
  font-size: 10px;
}
.chart-source.success { color: var(--text-success); border-color: var(--border-success); background: var(--bg-success-tint); }
.chart-source.warning { color: var(--text-warning); border-color: var(--color-warning-800); background: var(--bg-warning-tint); }
.chart-source.error { color: var(--text-error); border-color: var(--color-error-800); background: var(--bg-error-tint); }
.chart-source.muted { color: var(--text-tertiary); }
.portfolio-live-badge {
  display: inline-flex;
  align-items: center;
  gap: 8px;
  min-width: 0;
  padding: 2px 6px;
  border: 1px solid var(--border-default);
  background: var(--bg-container-2);
  border-radius: 4px;
}
.chart-indicator-fill { fill: oklch(from var(--color-info-500) l c h / 0.07); }
.chart-indicator-top,
.chart-indicator-bottom {
  fill: none;
  stroke: var(--color-info-400);
  stroke-width: 1.1;
  stroke-dasharray: 4 3;
}
.chart-indicator-mid {
  fill: none;
  stroke: var(--text-tertiary);
  stroke-width: 0.8;
  stroke-dasharray: 2 5;
  opacity: 0.7;
}
.chart-tp-line { stroke: var(--color-success-500); stroke-dasharray: 3 3; stroke-width: 1; }
.chart-order-line { stroke: var(--color-warning-500); stroke-dasharray: 3 3; stroke-width: 1; }
.chart-position-line { stroke: var(--text-secondary); stroke-dasharray: 2 6; stroke-width: 1; }
.chart-inline-price-label rect {
  fill: oklch(from var(--bg-container-2) l c h / 0.92);
  stroke: var(--border-strong);
  stroke-width: 1;
}
.chart-inline-price-label text {
  fill: var(--text-primary);
  font-family: var(--font-mono);
  font-size: 10px;
  font-weight: 700;
}
.chart-inline-price-label-connector {
  stroke: var(--border-strong);
  stroke-width: 1;
  stroke-dasharray: 2 3;
}
.chart-position-inline-label rect {
  fill: oklch(from var(--bg-neutral-press) l c h / 0.94);
}
.chart-buy-marker { fill: var(--color-success-500); }
.chart-sell-marker { fill: var(--color-warning-500); }
.chart-layer-switches {
  display: inline-flex;
  align-items: center;
  flex-wrap: wrap;
  gap: 4px;
}
/* Чип-слой графика приведён к виду тега из макета (desktop.dc.html ~152): пилюля
   с цветным штрихом-свотчем и названием, mono-шрифт. Это по-прежнему интерактивный
   тумблер ChartLayerToggle (button + onClick), просто без видимого «рубильника»:
   включённый слой — обычный тег, выключенный — приглушённый (opacity .5), как
   неактивная «История» в макете. */
.chart-layer-toggle {
  display: inline-flex;
  align-items: center;
  gap: 5px;
  padding: 2px 8px;
  border: 1px solid var(--border-default);
  border-radius: 999px;
  background: var(--bg-neutral);
  color: var(--text-secondary);
  font-size: 11px;
  font-weight: 400;
  font-family: var(--font-mono);
  line-height: 1.5;
  cursor: pointer;
  opacity: .5;
}
.chart-layer-toggle.on {
  opacity: 1;
  color: var(--text-secondary);
  border-color: var(--border-default);
}
/* Видимый рубильник макету в этой полосе не нужен — слой переключается кликом по чипу. */
.chart-layer-toggle .switch { display: none; }
.chart-layer-dot {
  width: 9px;
  height: 2px;
  border-radius: 0;
  background: var(--text-tertiary);
}
.chart-layer-toggle.limit .chart-layer-dot { background: var(--color-warning-500); }
.chart-layer-toggle.tp .chart-layer-dot { background: var(--color-success-500); }
.chart-layer-toggle.orders .chart-layer-dot { background: var(--text-tertiary); }
.chart-layer-toggle.history .chart-layer-dot { background: var(--color-info-400); }
.chart-live-order-marker circle {
  fill: var(--bg-neutral-press);
  stroke: var(--text-primary);
  stroke-width: 1;
}
.chart-live-order-marker text,
.chart-trade-history-line text,
.chart-trade-history-marker text {
  fill: var(--text-primary);
  font-family: var(--font-mono);
  font-size: 10px;
}
.chart-trade-history-line line {
  stroke: var(--text-primary);
  stroke-width: 1;
  stroke-dasharray: 4 4;
  opacity: 0.75;
}
.chart-trade-history-marker path {
  fill: var(--color-info-400);
  stroke: var(--bg-container);
  stroke-width: 1;
}
.chart-trade-history-marker.sell path {
  fill: var(--color-error-500);
}
.chart-tester-line line {
  stroke: var(--color-warning-500);
  stroke-width: 1;
  stroke-dasharray: 4 3;
  opacity: 0.86;
}
.chart-tester-line.closed line {
  stroke: var(--color-success-500);
}
.chart-tester-line text,
.chart-tester-marker text {
  fill: var(--text-primary);
  font-family: var(--font-mono);
  font-size: 9px;
  font-weight: 700;
  paint-order: stroke;
  stroke: var(--bg-container);
  stroke-width: 2px;
}
.chart-tester-marker path {
  fill: var(--color-warning-500);
  stroke: var(--bg-container);
  stroke-width: 1px;
}
.chart-tester-marker.sell path {
  fill: var(--color-success-500);
}
.chart-crosshair line {
  stroke: var(--text-secondary);
  stroke-width: 0.7;
  stroke-dasharray: 3 3;
  opacity: 0.72;
}
.chart-crosshair rect {
  fill: var(--bg-neutral-press);
  stroke: var(--border-strong);
  stroke-width: 1;
}
.chart-crosshair text {
  fill: var(--text-primary);
  font-family: var(--font-mono);
  font-size: 10px;
  font-weight: 600;
}
.chart-ruler line {
  stroke: var(--color-warning-500);
  stroke-width: 1.2;
}
.chart-ruler circle {
  fill: var(--color-warning-500);
  stroke: var(--bg-container);
  stroke-width: 1.5;
}
.chart-ruler rect {
  fill: var(--bg-container-2);
  stroke: var(--color-warning-500);
  stroke-width: 1;
}
.chart-ruler text {
  fill: var(--text-primary);
  font-family: var(--font-mono);
  font-size: 10px;
  font-weight: 700;
}

/* Focus reveal animation */
@keyframes fadein { from { opacity: 0; transform: translateY(4px); } to { opacity: 1; transform: none; } }
.fadein { animation: fadein .18s ease-out; }
/* То же появление, но БЕЗ transform — чтобы элемент не становился containing-block для
   fixed-потомков (см. мобильную полноэкранную панель «Настройки бумаги», ПУНКТ 4). */
@keyframes fadein-noshift { from { opacity: 0; } to { opacity: 1; } }
@keyframes row-new-flash {
  0% {
    background: color-mix(in srgb, var(--bg-brand) 42%, transparent);
    box-shadow: inset 3px 0 0 var(--bg-brand);
  }
  55% {
    background: color-mix(in srgb, var(--bg-brand) 24%, transparent);
    box-shadow: inset 3px 0 0 var(--bg-brand);
  }
  100% {
    background: transparent;
    box-shadow: inset 0 0 0 transparent;
  }
}
@keyframes testerApplyFlash {
  0% {
    background: color-mix(in srgb, var(--bg-success) 34%, transparent);
    box-shadow:
      inset 3px 0 0 var(--border-success),
      inset 0 0 0 999px color-mix(in srgb, var(--bg-success) 8%, transparent);
  }
  45% {
    background: color-mix(in srgb, var(--bg-success) 22%, transparent);
    box-shadow:
      inset 3px 0 0 var(--border-success),
      inset 0 0 0 999px color-mix(in srgb, var(--bg-success) 4%, transparent);
  }
  100% {
    background: transparent;
    box-shadow:
      inset 2px 0 0 color-mix(in srgb, var(--border-success) 72%, transparent),
      inset 0 0 0 999px transparent;
  }
}

/* Expandable orders row */
.expand-row td { padding: 0; }
.chev-toggle {
  display: inline-grid; place-items: center;
  width: 18px; height: 18px;
  color: var(--text-tertiary);
  transition: transform .15s, color .12s;
}
.chev-toggle.open { transform: rotate(180deg); color: var(--text-brand); }

/* Мобильные карточки «Заявки и ордера» — в DOM всегда, но видны только на ≤640 (см. медиа ниже).
   На десктопе показывается .orders-grid (таблица), карточки скрыты. */
.bond-orders-cards { display: none; }
.bond-orders-settings-btn { display: none; }
.orders-grid {
  display: grid;
  /* 10 столбцов: # | Тип | Сетка / ордер | Цена | Лот | TP | PnL | Состояние | Время | действие.
     PnL под «+1 234 ₽ · +1.23%». Последняя колонка 128px (макет desktop.dc.html ~163) — под
     текстовые кнопки действий «Изм. TP» / «✕ Отменить», а не только иконку. */
  grid-template-columns: 28px 76px minmax(96px, 1fr) 84px 64px 84px minmax(104px, 1fr) minmax(120px, 1.4fr) minmax(96px, 1fr) 128px;
  gap: 0 8px;
  align-items: center;
  background: var(--bg-container);
  border: 0;
  border-radius: 0;
  overflow: hidden;
}
.orders-grid-head {
  display: contents;
}
.orders-grid-head > span {
  font-size: 9px;
  text-transform: uppercase;
  letter-spacing: .04em;
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  padding: 6px 4px;
  background: var(--bg-neutral);
  border-bottom: 1px solid var(--border-default);
}
.orders-grid-head > span.num { text-align: right; }
/* Хвостовая колонка (под кнопку удаления) в заголовке пустая — снимаем фон/линию,
   чтобы не висел «артефактный» тёмный прямоугольник справа от «Время». */
.orders-grid-head > span:last-child { background: transparent; border-bottom-color: transparent; }
.orders-grid-row { display: contents; }
/* Плашка свободной (ещё не открытой) сетки — приглушённая, отделена пунктиром сверху. */
.orders-grid-row.grid-slot-row > span { opacity: 0.75; border-top: 1px dashed var(--border-default); }
.orders-grid-row > span {
  padding: 6px 4px;
  font-size: 11px;
  border-bottom: 1px solid var(--border-default);
}
.orders-grid-row.tp-saved-flash > span {
  animation: tpSavedRowFlash 1.3s ease-out;
}
.orders-grid-row.order-row-enter > span {
  animation: orderRowEnter .65s ease-out;
}
.orders-grid-row.order-row-update > span {
  animation: orderRowUpdate .9s ease-out;
}
.orders-grid-row.order-row-shrink > span {
  animation: orderRowShrink .95s ease-out;
}
.orders-grid-row.order-row-leave > span {
  animation: orderRowLeave .85s ease forwards;
  pointer-events: none;
}
.orders-grid-row:last-child > span { border-bottom: 0; }
.orders-grid-row > span.num { text-align: right; }
.tp-inline-input {
  width: 64px;
  height: 24px;
  padding: 2px 6px;
  text-align: right;
  font-size: 11px;
}

.orders-params {
  display: grid;
  grid-template-columns: 1fr auto;
  gap: 4px 12px;
  font-size: 11px;
  background: var(--bg-container);
  border: 1px solid var(--border-default);
  border-radius: 6px;
  padding: 10px 12px;
}
.orders-params-label { color: var(--text-tertiary); font-family: var(--font-mono); font-size: 10px; text-transform: uppercase; letter-spacing: .04em; align-self: center; }
.orders-params-value { color: var(--text-primary); font-family: var(--font-mono); font-weight: 500; text-align: right; }

@keyframes tpSavedRowFlash {
  0% {
    background: oklch(from var(--color-success-500) l c h / 0.34);
    box-shadow: inset 0 0 0 1px var(--border-success);
  }
  45% {
    background: oklch(from var(--color-success-500) l c h / 0.18);
    box-shadow: inset 0 0 0 1px oklch(from var(--color-success-500) l c h / 0.45);
  }
  100% {
    background: transparent;
    box-shadow: inset 0 0 0 1px transparent;
  }
}

@keyframes orderRowEnter {
  0% {
    background: oklch(from var(--color-brand-500) l c h / 0.30);
    transform: translateY(-4px);
    opacity: .15;
  }
  45% {
    background: oklch(from var(--color-brand-500) l c h / 0.16);
    opacity: 1;
  }
  100% {
    background: transparent;
    transform: translateY(0);
    opacity: 1;
  }
}

@keyframes orderRowUpdate {
  0% {
    background: oklch(from var(--color-info-500) l c h / 0.26);
    box-shadow: inset 0 0 0 1px oklch(from var(--color-info-500) l c h / 0.35);
  }
  100% {
    background: transparent;
    box-shadow: inset 0 0 0 1px transparent;
  }
}

@keyframes orderRowShrink {
  0% {
    background: oklch(from var(--color-warning-500) l c h / 0.28);
    box-shadow: inset 0 0 0 1px oklch(from var(--color-warning-500) l c h / 0.45);
  }
  100% {
    background: transparent;
    box-shadow: inset 0 0 0 1px transparent;
  }
}

@keyframes orderRowLeave {
  0% {
    background: oklch(from var(--color-success-500) l c h / 0.28);
    opacity: 1;
    transform: translateX(0);
  }
  100% {
    background: transparent;
    opacity: 0;
    transform: translateX(22px);
  }
}

/* Group tabs (Bonds screen) */
.group-tab {
  display: inline-flex;
  align-items: center;
  gap: 6px;
  padding: 8px 12px;
  font-size: 12px;
  color: var(--text-secondary);
  cursor: pointer;
  border-bottom: 2px solid transparent;
  margin-bottom: -1px;
  white-space: nowrap;
  transition: color .12s, background .12s;
}
.group-tab:hover { color: var(--text-primary); background: var(--bg-neutral); }
.group-tab.active {
  color: var(--text-primary);
  border-bottom-color: var(--border-brand);
  background: var(--bg-brand-tint);
}
.group-tab.dragging {
  opacity: .65;
}
/* Дропалка вкладки группы — тот же размер/иконка/ховер, что и у дропалки строки (.priority-drag),
   чтобы «потяни» читалось одинаково и наверху (группы), и в таблице (строки). */
.group-tab-drag {
  display: inline-grid;
  place-items: center;
  width: 20px;
  height: 20px;
  border-radius: 4px;
  color: var(--text-tertiary);
  cursor: grab;
  transition: background .12s, color .12s;
  -webkit-user-drag: element;
  user-select: none;
  touch-action: none; /* иначе на тач палец скроллит список, а не тащит группу */
}
.group-tab-drag:hover {
  background: var(--bg-neutral-hover);
  color: var(--text-brand);
}
.group-tab-drag:active { cursor: grabbing; }
/* Номер приоритета группы: тонкий мьютед-префикс (макет даёт чистые вкладки drag·имя·счёт·⚙;
   цветную точку убрали, номер оставили — он = реальная позиция в очереди групп, которую
   соблюдает планировщик). Читается как тихая мета, не конкурирует с именем группы. */
.group-tab-priority {
  min-width: 12px;
  text-align: center;
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  font-weight: 600;
  font-size: 10px;
}
/* (Цветная точка группы .group-tab-dot убрана — вкладки приведены к чистому виду макета
   drag·имя·счёт·⚙; идентичность группы несёт имя, ранг — мьютед-номер .group-tab-priority.) */
.group-tab-count {
  display: inline-grid; place-items: center;
  min-width: 18px; height: 16px;
  padding: 0 5px;
  background: var(--bg-neutral);
  color: var(--text-tertiary);
  border-radius: 999px;
  font-size: 10px;
  font-family: var(--font-mono);
  font-weight: 600;
}
.group-tab.active .group-tab-count {
  background: var(--bg-brand);
  color: var(--text-on-brand);
}
.group-tab-warn {
  display: inline-grid; place-items: center;
  min-width: 16px; height: 16px;
  padding: 0 4px;
  background: var(--bg-warning-tint);
  color: var(--text-warning);
  border-radius: 999px;
  font-size: 10px;
  font-family: var(--font-mono);
  font-weight: 600;
}
/* Глиф ⚙ (U+2699) как в макете — монохромная текстовая шестерёнка, не цветное эмодзи. */
.group-tab-settings {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  font-size: 13px;
  line-height: 1;
  color: var(--text-secondary);
  font-variant-emoji: text;
  opacity: 1;
  transition: opacity .12s, color .12s;
  cursor: pointer;
  margin-left: 2px;
}
.group-tab-settings:hover { opacity: 1; color: var(--text-secondary); }
/* Шестерёнка рендерится на всех вкладках; на неактивных — приглушена, ховер возвращает яркость. */
.group-tab-settings.dim { opacity: .55; }
.group-tab-settings.dim:hover { opacity: 1; color: var(--text-secondary); }

/* Group chip — clickable, with color dot */
.group-chip {
  display: inline-flex;
  align-items: center;
  gap: 5px;
  padding: 3px 8px 3px 6px;
  background: var(--bg-neutral);
  border: 1px solid var(--border-default);
  border-radius: 4px;
  color: var(--text-secondary);
  font-size: 11px;
  font-family: inherit;
  cursor: pointer;
  transition: background .12s, border-color .12s, color .12s;
  white-space: nowrap;
  max-width: 180px;
}
.group-chip:hover { background: var(--bg-neutral-hover); border-color: var(--border-brand); color: var(--text-primary); }
.group-chip-dot { width: 6px; height: 6px; border-radius: 999px; flex-shrink: 0; }
.group-chip-label { overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }

/* Auto pill — A indicator */
.auto-pill {
  padding: 0;
  display: inline-grid;
  place-items: center;
  width: 22px; height: 22px;
  border-radius: 4px;
  background: var(--bg-neutral);
  color: var(--text-secondary);
  font-size: 10px;
  font-weight: 600;
  font-family: var(--font-mono);
  border: 1px solid var(--border-strong);
  cursor: pointer;
}
.auto-pill.on {
  background: var(--bg-success-tint);
  color: var(--text-success);
  border-color: var(--border-success);
}
.auto-pill:disabled {
  cursor: default;
  opacity: .65;
}

/* Link-style eyebrow */
.eyebrow.link {
  cursor: pointer;
  transition: color .12s;
  text-decoration: underline dotted;
  text-underline-offset: 2px;
}
.eyebrow.link:hover { color: var(--text-brand); }

/* Bond list card (left rail of Bonds screen) */
.bond-card {
  display: block;
  width: 100%;
  text-align: left;
  padding: 10px 12px;
  background: transparent;
  border: 0;
  border-bottom: 1px solid var(--border-default);
  cursor: pointer;
  font-family: inherit;
  color: inherit;
  transition: background .12s;
}
.bond-card:hover { background: var(--bg-neutral); }
.bond-card.active {
  background: var(--bg-brand-tint);
  border-left: 2px solid var(--border-brand);
  padding-left: 10px;
}

.priority-cell {
  /* НЕ display:flex на <td>! Флекс-ячейка перестаёт тянуться на высоту строки (становится
     высотой контента ~20px), и левый акцент-полоса + фон активной строки покрывают только
     верх — низ высокой 2-строчной строки даёт тёмный прямоугольник без акцента (тот самый
     «артефакт»). Держим нормальную table-cell — она тянется на всю высоту строки. Флекс-
     раскладку контента (дропалка·номер·точка) переносим на внутреннюю обёртку. */
  vertical-align: middle;
  color: var(--text-tertiary);
  white-space: nowrap;
}
.priority-cell-inner {
  display: flex;
  align-items: center;
  gap: 6px;
}
/* Статус-точка строки: компактный аккуратный пип (цвет = семантика статуса), сведённое
   свечение — чтобы не превращалась в тяжёлую кляксу и не наезжала на номер/дропалку.
   Цвет согласован с полосой-акцентом строки (--row-accent) и колонкой «Статус». */
.priority-cell .dot {
  width: 8px; height: 8px;
  box-shadow: 0 0 0 2px var(--bg-container);
}
.priority-drag {
  display: inline-grid;
  place-items: center;
  width: 20px;
  height: 20px;
  border-radius: 4px;
  color: var(--text-secondary);
  cursor: grab;
  transition: background .12s, color .12s;
  -webkit-user-drag: element;
  user-select: none;
  touch-action: none; /* иначе на тач палец скроллит список, а не тащит строку */
}
.priority-drag:hover {
  background: var(--bg-neutral-hover);
  color: var(--text-brand);
}
.priority-drag:active { cursor: grabbing; }
.priority-drag.disabled {
  cursor: default;
  opacity: .35;
}
body.priority-dragging,
body.priority-dragging * {
  cursor: grabbing !important;
  touch-action: none; /* пока тащим — страница/список не скроллятся под пальцем */
}
tr.priority-drop-target {
  outline: 1px solid var(--border-brand);
  outline-offset: -1px;
  background: var(--bg-brand-tint);
}
.priority-number {
  min-width: 18px;
  color: var(--text-secondary);
  font-family: var(--font-mono);
  font-weight: 700;
  font-size: 11px;
}

/* Multi-select chips (ratings, types) */
.rating-chip {
  display: inline-flex;
  align-items: center;
  padding: 3px 8px;
  border: 1px solid var(--border-default);
  background: var(--bg-container);
  color: var(--text-secondary);
  border-radius: 999px;
  font-family: var(--font-mono);
  font-size: 11px;
  font-weight: 500;
  cursor: pointer;
  transition: background .12s, border-color .12s, color .12s;
  white-space: nowrap;
}
.rating-chip:hover { background: var(--bg-neutral-hover); border-color: var(--border-strong); color: var(--text-primary); }
.rating-chip[data-active="true"] {
  background: var(--bg-brand-tint);
  border-color: var(--border-brand);
  color: var(--text-brand);
}

/* Modal */
.scrim { position: fixed; inset: 0; background: var(--bg-overlay); backdrop-filter: blur(2px); display: grid; place-items: center; z-index: 100; }
.modal { background: var(--bg-elevated); border: 1px solid var(--border-default); border-radius: 8px; box-shadow: var(--shadow-modal); min-width: 480px; max-width: 92vw; max-height: 90vh; overflow: auto; }
.modal-head {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 14px;
  padding: 16px 18px;
  border-bottom: 1px solid var(--border-default);
}
.modal-head h2 {
  margin: 0;
  font-size: 18px;
  line-height: 1.2;
}
.modal-body {
  padding: 16px 18px;
}
.modal-foot {
  display: flex;
  justify-content: flex-end;
  align-items: center;
  gap: 8px;
  padding: 14px 18px;
  border-top: 1px solid var(--border-default);
}
.group-action-modal {
  width: min(560px, calc(100vw - 32px));
  min-width: 0;
  overflow: hidden;
}
.correction-transfer-modal {
  width: min(520px, calc(100vw - 48px));
  min-width: 0;
}
.callout {
  display: grid;
  grid-template-columns: auto minmax(0, 1fr);
  gap: 10px;
  align-items: start;
  padding: 12px;
  border: 1px solid var(--border-info);
  background: var(--bg-info-tint);
  color: var(--text-info);
  border-radius: 6px;
}
.callout.warn {
  border-color: var(--border-warning);
  background: var(--bg-warning-tint);
  color: var(--text-warning);
}
.callout.error {
  border-color: var(--border-error);
  background: var(--bg-error-tint);
  color: var(--text-error);
}
.grid-2 {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: 10px;
}
.stat-mini {
  display: flex;
  flex-direction: column;
  gap: 4px;
  padding: 10px 12px;
  border: 1px solid var(--border-default);
  background: var(--bg-container);
  border-radius: 6px;
}
.stat-mini span {
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  font-size: 10px;
  text-transform: uppercase;
}
.stat-mini b {
  font-size: 18px;
}
.add-account-modal {
  width: min(640px, 94vw);
}
/* Модалка замены API-токена — компактнее формы добавления счёта. */
.replace-token-modal {
  width: min(440px, 94vw);
  min-width: 0;
  display: flex;
  flex-direction: column;
}
.replace-token-modal > .block-head { padding: 14px; border-bottom: 1px solid var(--border-default); }
/* Строка статуса токена на карточке счёта: длинный текст-тег может переноситься (узкая карточка). */
.accounts-screen .acct-card-token > .tag { white-space: normal; }
/* Биллинг (Фаза 3a): строка статуса лицензии/сервера на карточке счёта (тип · сервер · до даты).
   Раскладка вынесена из инлайнов в класс (этот заход владеет app.css). Десктоп: теги в ряд, перенос. */
.accounts-screen .acct-card-license {
  padding: 0 14px 12px;
  display: flex;
  flex-wrap: wrap;
  gap: 6px;
  align-items: center;
}
.accounts-screen .acct-card-license-title { margin-right: 2px; }
.accounts-screen .acct-card-license-until {
  display: inline-flex;
  align-items: center;
  gap: 3px;
  font-size: 11px;
  color: var(--text-tertiary);
}
.add-account-modal > .block-head,
.add-account-modal > .row {
  flex-shrink: 0;
}
.add-account-body {
  display: flex;
  flex-direction: column;
  gap: 16px;
  padding: 16px;
  overflow-y: auto;
  flex: 1;
  min-height: 0;
}
.add-account-steps {
  display: grid;
  gap: 10px;
}
.add-account-steps > div {
  display: grid;
  grid-template-columns: auto minmax(0, 1fr);
  gap: 10px;
  align-items: start;
}
.add-account-stepn {
  flex: 0 0 auto;
  width: 22px;
  height: 22px;
  border-radius: 50%;
  background: var(--bg-brand);
  color: var(--text-on-brand);
  display: grid;
  place-items: center;
  font-size: 11px;
  font-weight: 700;
  font-family: var(--font-mono);
}
.add-account-step-title {
  font-size: 13px;
  font-weight: 600;
  color: var(--text-primary);
}
.add-account-step-link {
  display: inline-flex;
  align-items: center;
  gap: 4px;
  font-size: 12px;
  color: var(--text-brand);
  text-decoration: none;
  width: fit-content;
}
.add-account-step-link:hover { text-decoration: underline; }
/* [NEW] Встроенный видео-плеер 16:9 — плейсхолдер видеоинструкции (источник позже, data-video-src) */
.add-account-video {
  position: relative;
  width: 100%;
  aspect-ratio: 16 / 9;
  border: 1px solid var(--border-strong);
  border-radius: 10px;
  overflow: hidden;
  background: linear-gradient(135deg, #11161a, #1b2228);
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  gap: 8px;
  cursor: pointer;
}
.add-account-video-play {
  width: 56px;
  height: 56px;
  border-radius: 50%;
  /* читаемый play-кружок: бренд-тинт + светлое кольцо, чтобы блок ЧИТАЛСЯ как видео,
     а не как пустая чёрная дыра (filled-треугольник задаётся в JSX через stroke={false}) */
  background: var(--bg-brand-tint, rgba(255, 255, 255, 0.12));
  box-shadow: 0 0 0 1px rgba(255, 255, 255, 0.18), 0 6px 18px rgba(0, 0, 0, 0.35);
  display: grid;
  place-items: center;
  color: var(--text-on-brand, #fff);
  padding-left: 3px; /* оптическая центровка треугольника */
  backdrop-filter: blur(4px);
}
.add-account-video:hover .add-account-video-play {
  background: var(--bg-brand, rgba(255, 255, 255, 0.22));
}
.add-account-video-cap {
  font-size: 11px;
  color: var(--text-tertiary);
}
.add-account-form {
  display: grid;
  grid-template-columns: 1fr;
  gap: 12px;
}
.add-account-preview {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  padding: 10px 12px;
  border: 1px solid var(--border-default);
  border-radius: 4px;
  background: var(--bg-neutral);
}
.account-review-state {
  box-sizing: border-box;
  min-width: 0;
  min-height: 220px;
  display: grid;
  place-items: center;
  align-content: center;
  gap: 10px;
  padding: 24px;
  text-align: center;
  color: var(--text-info);
}
.account-review-state.rejected {
  color: var(--text-error);
}
/* Заголовок/текст инфо-состояния переносятся (глобальный .h2 — nowrap),
   иначе длинный текст «Ожидание рассмотрения…»/причина отказа вылезает за рамку карточки */
.account-review-state .h2 {
  max-width: 100%;
  min-width: 0;
  white-space: normal;
  word-break: break-word;
  overflow-wrap: anywhere;
}
.account-review-state .muted {
  max-width: min(320px, 100%);
  min-width: 0;
  white-space: normal;
  word-break: break-word;
  overflow-wrap: anywhere;
}
.account-delete-modal,
.reject-account-modal {
  width: min(680px, 94vw);
}

.instrument-picker-modal {
  width: min(1500px, 96vw);
  height: min(860px, 92vh);
  display: flex;
  flex-direction: column;
}
.instrument-picker-toolbar {
  display: grid;
  gap: 10px;
  padding: 14px 16px;
  border-top: 1px solid var(--border-default);
  border-bottom: 1px solid var(--border-default);
}
.instrument-picker-table-wrap {
  flex: 1;
  min-height: 0;
  overflow: auto;
}
.instrument-picker-content {
  flex: 1;
  min-height: 0;
  display: grid;
  grid-template-columns: minmax(0, 1fr);
  background: var(--bg-container);
}
.instrument-picker-content.with-preview {
  grid-template-columns: minmax(360px, 1fr) minmax(760px, 2fr);
}
.instrument-picker-table {
  min-width: 1420px;
}
.instrument-picker-table.compact {
  min-width: 760px;
}
.instrument-picker-table.compact th:first-child,
.instrument-picker-table.compact td:first-child {
  min-width: 280px;
}
.instrument-picker-table thead th {
  position: sticky;
  top: 0;
  z-index: 1;
  background: var(--bg-elevated);
}
.instrument-picker-table td {
  white-space: nowrap;
}
.instrument-picker-table tbody tr {
  cursor: pointer;
}
.instrument-picker-table tbody tr:hover td {
  background: var(--bg-neutral-hover);
}
.instrument-picker-table tbody tr.selected td {
  background: oklch(from var(--bg-brand-tint) l c h / 0.92);
  color: var(--text-primary);
}
.instrument-picker-table tbody tr.selected td:first-child {
  box-shadow: inset 4px 0 0 var(--color-brand-400);
}
.instrument-picker-table .sec {
  max-width: 360px;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.instrument-preview-panel {
  min-width: 0;
  border-left: 1px solid var(--border-default);
  background: var(--bg-elevated);
  display: grid;
  grid-template-rows: auto minmax(0, 1fr);
  overflow: hidden;
}
.instrument-preview-head {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  padding: 10px 14px;
  border-bottom: 1px solid var(--border-default);
}
.instrument-preview-head.compact {
  min-height: 54px;
  padding: 8px 12px;
}
.instrument-preview-head h3 {
  margin: 0;
  color: var(--text-primary);
  font-size: 14px;
  line-height: 1.25;
}
.instrument-preview-title {
  min-width: 0;
  display: grid;
  gap: 2px;
  flex: 1;
}
.instrument-preview-title b {
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  color: var(--text-primary);
  font-size: 15px;
}
.instrument-preview-chart-head {
  display: flex;
  align-items: center;
  justify-content: flex-start;
  gap: 10px;
  padding: 10px 14px 8px;
}
.instrument-preview-chart {
  min-height: 0;
  margin: 10px 14px 14px;
  border: 1px solid var(--border-default);
  border-radius: 6px;
  overflow: hidden;
  background: var(--bg-container);
}
.instrument-preview-empty {
  height: 100%;
  min-height: 220px;
  display: grid;
  place-content: center;
  gap: 8px;
  color: var(--text-tertiary);
  text-align: center;
}
.instrument-preview-empty b {
  color: var(--text-primary);
}
.instrument-preview-chart .chart-floating-tools {
  display: none;
}
.instrument-preview-chart .chart-indicator-fill {
  fill: oklch(from var(--color-info-500) l c h / 0.16);
}
.instrument-preview-chart .chart-indicator-top,
.instrument-preview-chart .chart-indicator-bottom {
  stroke: var(--color-info-300);
  stroke-width: 2;
  stroke-dasharray: 6 4;
}
.instrument-preview-chart .chart-indicator-mid {
  stroke: var(--color-success-500);
  stroke-width: 1.35;
}
.group-candidate-chart {
  border-bottom: 1px solid var(--border-default);
  background: var(--bg-elevated);
  overflow: hidden;
}
.group-candidate-chart.slide-down {
  animation: groupChartSlide .16s ease-out;
}
.group-candidate-chart-head {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  padding: 12px 16px 8px;
}
.group-candidate-chart-body {
  height: 300px;
  margin: 0 16px 14px;
  border: 1px solid var(--border-default);
  border-radius: 6px;
  overflow: hidden;
  background: var(--bg-container);
}
.group-candidate-chart-body .chart-floating-tools {
  display: none;
}
.group-candidate-chart-body .chart-indicator-fill {
  fill: oklch(from var(--color-info-500) l c h / 0.16);
}
.group-candidate-chart-body .chart-indicator-top,
.group-candidate-chart-body .chart-indicator-bottom {
  stroke: var(--color-info-300);
  stroke-width: 2;
  stroke-dasharray: 6 4;
}
.group-candidate-chart-body .chart-indicator-mid {
  stroke: var(--color-success-500);
  stroke-width: 1.35;
}
@keyframes groupChartSlide {
  from { opacity: 0; transform: translateY(-8px); }
  to { opacity: 1; transform: translateY(0); }
}
.duration-filter {
  position: relative;
}
.duration-filter-button {
  min-width: 136px;
  justify-content: space-between;
  gap: 8px;
}
.duration-filter-button.active {
  border-color: var(--border-brand);
  color: var(--text-brand);
}
.duration-filter-divider {
  width: 1px;
  height: 18px;
  margin-left: auto;
  background: var(--border-strong);
}
.duration-filter-menu {
  position: absolute;
  top: calc(100% + 6px);
  right: 0;
  z-index: 160;
  display: flex;
  flex-direction: column;
  min-width: 150px;
  padding: 6px;
  border: 1px solid var(--border-default);
  border-radius: 7px;
  background: var(--bg-elevated);
  box-shadow: 0 18px 42px rgba(0, 0, 0, .35);
}
.duration-filter-menu button {
  display: block;
  width: 100%;
  min-height: 32px;
  padding: 6px 10px;
  border: 0;
  border-radius: 4px;
  background: transparent;
  color: var(--text-primary);
  font: inherit;
  font-weight: 700;
  text-align: left;
  cursor: pointer;
}
.duration-filter-menu button:hover,
.duration-filter-menu button.active {
  background: var(--bg-neutral-hover);
  color: var(--text-brand);
}
.instrument-picker-footer {
  display: flex;
  align-items: center;
  justify-content: flex-end;
  gap: 12px;
  min-height: 38px;
  padding: 6px 16px;
  border-top: 1px solid var(--border-default);
  background: var(--bg-elevated);
}
/* ===== Справочник · мобильная раскладка (эталон mobile.dc.html стр.229, 240).
   Десктоп split-view (≥641) НЕ трогаем. ===== */
.picker-scroll-hint { display: none; }
@media (max-width: 640px) {
  /* Модалка-боттомшит (.modal: width 100%, min-width 0 уже из дженерика) — режем
     любой горизонт-вылет внутреннего контента, чтобы шапка/тулбар/контент не
     «уезжали» влево, когда дочерний блок шире вьюпорта. */
  .instrument-picker-modal { overflow-x: hidden; }
  /* КОРНЕВАЯ ПРИЧИНА СДВИГА: модалка — flex-column, а её прямые дети (block-head,
     toolbar, content, footer) по умолчанию min-width:auto и НЕ ужимаются ниже
     min-content. Тулбар содержал Segmented с длинными лейблами («Фонды ликвидности»),
     который раздувал свой грид-трек до ~507px → весь столбец шире экрана, и flex
     центрировал переполненных детей с отрицательным left (контент уезжал влево,
     «×»/поиск/чипы обрезались). Фиксируем min-width:0 по всем прямым детям. */
  .instrument-picker-modal > .block-head,
  .instrument-picker-modal > .instrument-picker-toolbar,
  .instrument-picker-modal > .instrument-picker-content,
  .instrument-picker-modal > .instrument-picker-footer,
  .instrument-picker-modal > .reason { min-width: 0; }
  /* Тулбар: единственный грид-трек жёстко кап minmax(0,1fr) — иначе auto-трек
     растягивается под самый широкий Segmented и распирает модалку. */
  .instrument-picker-toolbar { min-width: 0; grid-template-columns: minmax(0, 1fr); }
  /* Segmented'ы (класс актива / вид облигаций) — длинные ряды кнопок: даём им
     ужиматься (min-width:0) и скроллиться вбок ВНУТРИ себя, чтобы не толкали
     ширину тулбара; кнопки не переносим, они скроллятся. */
  .instrument-picker-toolbar .seg {
    min-width: 0; max-width: 100%;
    overflow-x: auto; -webkit-overflow-scrolling: touch; scrollbar-width: none;
  }
  .instrument-picker-toolbar .seg::-webkit-scrollbar { display: none; }
  .instrument-picker-toolbar .seg button { flex: 0 0 auto; white-space: nowrap; }
  /* Ряд «вид облигаций» (Segmented + Фильтрация/Дюрация) — переносим вместо
     space-between по краям, и каждый блок ужимается. */
  .instrument-picker-toolbar > .row { flex-wrap: wrap; row-gap: 8px; min-width: 0; }
  .instrument-picker-toolbar > .row > .seg { flex: 1 1 100%; }
  /* Поле поиска: flex:1 ужимается ниже контента только при min-width:0. */
  .instrument-picker-toolbar > .row.gap-8 { flex-wrap: nowrap; }
  .instrument-picker-toolbar > .row.gap-8 > .input { min-width: 0; }

  /* min-width:0 по цепочке: иначе широкая таблица (min-width:1420) распирает грид-трек
     и вся страница/модалка вылезает вбок. Фиксируем — трек держит ширину экрана,
     таблица скроллит ВНУТРИ .instrument-picker-table-wrap. */
  .instrument-picker-content { min-width: 0; }
  .instrument-picker-content,
  .instrument-picker-content.with-preview {
    grid-template-columns: minmax(0, 1fr);
    /* контент-колонка сама становится вертикальным скроллером, чтобы превью могло
       залипнуть сверху, а таблица — скроллиться вбок под ним */
    overflow-y: auto; -webkit-overflow-scrolling: touch;
  }
  .instrument-picker-table-wrap {
    min-width: 0; flex: none;
    overflow-x: auto; overflow-y: visible; -webkit-overflow-scrolling: touch;
  }
  /* превью бумаги — закреплённая карточка НАД списком (эталон стр.229) */
  .instrument-picker-content.with-preview .instrument-preview-panel {
    order: -1; position: sticky; top: 0; z-index: 6;
    min-width: 0; max-height: 42vh;
    border-left: 0; border-bottom: 1px solid var(--border-default);
    background: var(--bg-app);
  }
  .instrument-preview-panel .instrument-preview-chart { margin: 8px 12px 12px; }
  /* Шапка превью (тикер + таймфреймы + «Скрыть») — flex space-between; на узком
     экране сумма детей чуть шире панели и давала микро-вылет. Ужимаем: заголовок
     min-width:0 с эллипсисом, ряд таймфреймов скроллится вбок внутри себя. */
  .instrument-preview-panel .instrument-preview-head { min-width: 0; }
  .instrument-preview-panel .instrument-preview-head .instrument-preview-title { min-width: 0; }
  .instrument-preview-panel .instrument-preview-head .seg {
    min-width: 0; overflow-x: auto; -webkit-overflow-scrolling: touch; scrollbar-width: none;
  }
  .instrument-preview-panel .instrument-preview-head .seg::-webkit-scrollbar { display: none; }
  .instrument-preview-panel .instrument-preview-head .seg button { flex: 0 0 auto; white-space: nowrap; }
  /* подсказка про горизонт-скролл под таблицей */
  .picker-scroll-hint { display: block; font-size: 10px; padding: 6px 14px 2px; }
}
.instrument-filter-scrim {
  z-index: 120;
}
.instrument-filter-modal {
  width: min(900px, 94vw);
}
.instrument-filter-body {
  display: grid;
  gap: 16px;
  padding: 16px;
  border-top: 1px solid var(--border-default);
}
.instrument-filter-grid {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: 14px 24px;
}
.instrument-filter-range {
  display: grid;
  gap: 6px;
}
.instrument-filter-range .input {
  min-width: 0;
}
.instrument-filter-section {
  display: grid;
  gap: 8px;
  padding-top: 12px;
  border-top: 1px solid var(--border-default);
}
.instrument-filter-footer {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  padding: 12px 16px;
  border-top: 1px solid var(--border-default);
  background: var(--bg-elevated);
}

.group-delete-modal {
  width: min(760px, 94vw);
}
.group-delete-body {
  display: grid;
  gap: 14px;
  padding: 16px;
}
.group-delete-footer {
  display: flex;
  align-items: center;
  justify-content: flex-end;
  gap: 8px;
  padding: 12px 16px;
  border-top: 1px solid var(--border-default);
  background: var(--bg-elevated);
}
.delete-preview {
  display: grid;
  gap: 8px;
  padding: 12px;
  border: 1px solid var(--border-default);
  border-radius: 6px;
  background: var(--bg-container);
}
.delete-preview .t {
  border: 1px solid var(--border-default);
  border-radius: 4px;
  overflow: hidden;
}
.delete-preview-empty {
  padding: 12px;
  border: 1px dashed var(--border-default);
  border-radius: 4px;
  color: var(--text-secondary);
  font-size: 12px;
}

.group-filter-row {
  display: grid;
  grid-template-columns: minmax(92px, 1fr) 154px;
  gap: 8px;
  align-items: center;
}

.group-filter-inputs {
  display: grid;
  grid-template-columns: 1fr 1fr;
  gap: 6px;
}

.group-filter-inputs .input {
  height: 34px;
  padding: 0 8px;
  text-align: right;
}

.t.compact th,
.t.compact td {
  padding: 7px 9px;
}

.portfolio-tester-modal {
  width: min(1500px, 96vw);
  max-height: 94vh;
}
.portfolio-tester-grid {
  display: grid;
  grid-template-columns: 450px minmax(0, 1fr);
  gap: 12px;
  padding: 12px;
}
.portfolio-run-summary {
  display: grid;
  grid-template-columns: 1fr;
  border-bottom: 1px solid var(--border-default);
}
.portfolio-run-summary .kpi {
  min-height: 86px;
  border-right: 0;
}
.portfolio-run-actions,
.portfolio-bond-toolbar {
  display: flex;
  gap: 8px;
  align-items: center;
  padding: 10px 12px;
  border-bottom: 1px solid var(--border-default);
}
.portfolio-bond-toolbar .input {
  flex: 1;
  min-width: 220px;
}
.portfolio-bond-list {
  max-height: 360px;
  overflow: auto;
}
.portfolio-bond-row {
  display: grid;
  grid-template-columns: minmax(220px, 1.3fr) minmax(130px, .8fr) 70px 58px 104px auto;
  align-items: center;
  gap: 10px;
  padding: 10px 12px;
  border-bottom: 1px solid var(--border-default);
}
.portfolio-bond-row:last-child { border-bottom: 0; }
.portfolio-bond-row.off {
  opacity: .48;
}
.portfolio-bond-meta {
  min-width: 0;
  color: var(--text-secondary);
  font-size: 11px;
  font-family: var(--font-mono);
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}
.portfolio-run-progress {
  flex: 1;
  min-width: 320px;
  max-width: 620px;
  display: grid;
  gap: 5px;
}
.portfolio-run-progress.idle {
  display: flex;
  justify-content: center;
}
.portfolio-run-progress-top {
  display: flex;
  align-items: center;
  gap: 8px;
  min-width: 0;
  color: var(--text-secondary);
  font-size: 11px;
}
.portfolio-run-progress-top b {
  min-width: 0;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  color: var(--text-primary);
}
.portfolio-run-progress-bar {
  height: 7px;
  overflow: hidden;
  border-radius: 999px;
  background: var(--color-neutral-900);
  border: 1px solid var(--border-default);
}
.portfolio-run-progress-bar i {
  display: block;
  height: 100%;
  min-width: 4px;
  border-radius: inherit;
  background: linear-gradient(90deg, var(--bg-brand), var(--color-success-500));
  transition: width .45s ease;
}
.portfolio-run-progress.running .portfolio-run-progress-bar i {
  background-size: 180% 100%;
  animation: portfolioProgressFlow 1.2s linear infinite;
}
.portfolio-run-progress-detail {
  min-height: 13px;
  color: var(--text-tertiary);
  font-size: 10px;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}
.portfolio-group-list {
  display: flex;
  flex-direction: column;
  max-height: 288px;
  overflow: auto;
}
.portfolio-group-row {
  display: grid;
  grid-template-columns: minmax(0, 1fr) auto;
  align-items: center;
  gap: 10px;
  padding: 10px 12px;
  border-bottom: 1px solid var(--border-default);
  color: var(--text-secondary);
  font-size: 12px;
}
.portfolio-group-row.active {
  background: var(--bg-brand-tint);
  color: var(--text-primary);
}
.portfolio-group-name {
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  font-weight: 700;
}
.empty {
  padding: 18px 12px;
  color: var(--text-tertiary);
  font-size: 12px;
}
.portfolio-summary {
  display: grid;
  grid-template-columns: repeat(4, minmax(140px, 1fr));
}
.portfolio-summary-placeholder {
  display: flex;
  align-items: center;
  justify-content: center;
  min-height: 112px;
  padding: 18px;
  color: var(--text-tertiary);
  font-size: 13px;
}
.portfolio-final-report {
  overflow: hidden;
  background: var(--bg-container);
}
.portfolio-final-status {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  padding: 12px 16px 8px;
}
.portfolio-run-ok {
  display: inline-flex;
  align-items: center;
  gap: 8px;
  color: var(--color-success-400);
  font-family: var(--font-mono);
  font-size: 12px;
}
.portfolio-run-ok::before {
  content: "";
  width: 6px;
  height: 6px;
  border-radius: 50%;
  background: currentColor;
}
.portfolio-final-head {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  padding: 12px 18px;
  border: 1px solid var(--border-default);
  border-left: 0;
  border-right: 0;
  background: var(--bg-elevated);
}
.portfolio-final-head h3 {
  margin: 0;
  font-size: 15px;
  line-height: 1.2;
  text-transform: uppercase;
}
.portfolio-final-kpis {
  display: grid;
  grid-template-columns: repeat(4, minmax(0, 1fr));
  border-bottom: 1px solid var(--border-default);
}
.portfolio-final-kpi {
  display: grid;
  align-content: start;
  gap: 7px;
  min-height: 104px;
  padding: 16px 18px;
  border-right: 1px solid var(--border-default);
}
.portfolio-final-kpi:last-child {
  border-right: 0;
}
.portfolio-final-kpi .label {
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  font-size: 12px;
  text-transform: uppercase;
}
.portfolio-final-kpi b {
  color: var(--text-primary);
  font-family: var(--font-mono);
  font-size: 26px;
  line-height: 1.05;
}
.portfolio-final-kpi small {
  color: var(--text-secondary);
  font-family: var(--font-mono);
  font-size: 12px;
}
.portfolio-final-kpi small b {
  color: var(--color-success-400);
  font-size: inherit;
}
.portfolio-final-grid {
  display: grid;
  grid-template-columns: 1fr 1fr 1fr;
  border-bottom: 1px solid var(--border-default);
}
.portfolio-final-col {
  display: grid;
  align-content: start;
  gap: 18px;
  min-height: 360px;
  padding: 16px 18px;
  border-right: 1px solid var(--border-default);
}
.portfolio-final-col:last-child {
  border-right: 0;
}
.portfolio-final-section {
  display: grid;
  gap: 10px;
}
.portfolio-final-section h4 {
  margin: 0 0 1px;
  padding-bottom: 6px;
  border-bottom: 1px dashed var(--border-default);
  color: var(--text-tertiary);
  font-family: var(--font-mono);
  font-size: 12px;
  font-weight: 500;
  text-transform: uppercase;
}
.portfolio-final-line {
  display: grid;
  grid-template-columns: minmax(0, 1fr) auto;
  align-items: baseline;
  gap: 12px;
  color: var(--text-secondary);
  font-size: 14px;
}
.portfolio-final-line b {
  color: var(--text-primary);
  font-family: var(--font-mono);
  font-size: 14px;
  font-weight: 600;
  text-align: right;
  white-space: nowrap;
}
.portfolio-final-line.strong span,
.portfolio-final-line.strong b {
  color: var(--text-primary);
  font-weight: 800;
}
.portfolio-final-line.highlight {
  margin: 0 -18px;
  padding: 8px 18px;
  background: rgba(148, 163, 184, .07);
  border-top: 1px solid var(--border-default);
}
.portfolio-final-footer {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  padding: 14px 18px;
  background: var(--bg-elevated);
}
.portfolio-report-wrap {
  max-height: 38vh;
  overflow: auto;
}
/* «Тестеры»: липкий первый столбец отчёта (инструмент/группа) при горизонтальном скролле широких таблиц. */
.portfolio-report-wrap table.t thead th:first-child { position: sticky; left: 0; z-index: 3; background: var(--bg-container); }
.portfolio-report-wrap table.t tbody td:first-child { position: sticky; left: 0; z-index: 2; background: var(--bg-app); }
.portfolio-report-wrap table.t tbody tr.tester-total-row td:first-child { background: var(--bg-container-2); }
/* Подсказка горизонт-скролла под отчётом — только мобайл (как .tester-scroll-hint в полном тестере). */
.portfolio-report-hint { display: none; }

@keyframes portfolioProgressFlow {
  0% { background-position: 0% 0; }
  100% { background-position: 180% 0; }
}

/* ===== ПОРТФЕЛЬНЫЙ ТЕСТЕР · мобайл (≤640px), эталон handoff/testers/mobile.dc.html.
   Десктоп (≥641) НЕ трогаем. Состав 1:1: модал → полноэкранный лист; выбор групп/бумаг →
   карточки .bcard со switch; 4 KPI сводки → 2 в ряд; 7 секций «Итоги портфеля» стопкой;
   вкладки отчёта → mchip-чипы (горизонт-скролл); таблицы отчёта → tbl-scroll + min-width +
   подсказка «← листай вправо». Без горизонт-вылета страницы. ===== */
@media (max-width: 640px) {
  /* Модал портфельного → полноэкранный лист (без center-scrim). */
  .scrim:has(> .portfolio-tester-modal) {
    place-items: stretch;
    padding: 0;
  }
  /* .scrim > … повышает специфичность над дженерик-боттомшитом `.modal` ниже (line ~2964),
     иначе портфельный модал получает скруглённый верх и max-height:88vh вместо полного листа. */
  .scrim > .portfolio-tester-modal {
    width: 100vw;
    min-width: 0;
    max-width: 100vw;
    /* 100dvh (как .shell/.modal.gr-modal/.ovr-screen выше) — динамическая высота вьюпорта:
       на iOS 100vh = БОЛЬШОЙ вьюпорт (без браузерного бара) и срезает низ листа; dvh честный.
       Безопасные зоны устройства (вырез/«чёлка» сверху, индикатор-home снизу) НЕ срезаем
       контентом листа — отступы под них несёт липкая шапка/футер ниже (env(safe-area-inset-*)),
       как у savebar'ов в этом файле. */
    max-height: 100dvh;
    height: 100dvh;
    border: 0;
    border-radius: 0;
    box-shadow: none;
    overflow-y: auto;
    overflow-x: hidden;
    -webkit-overflow-scrolling: touch;
  }
  /* Шапка модала: липкая сверху, чтобы «Закрыть» (×) всегда под рукой при длинном листе.
     padding-top += safe-area-inset-top — иначе заголовок и × уезжают под вырез/статус-бар
     устройства (на iPhone «прячутся за верхней панелью»). Inline padding шапки (14px 16px)
     перебиваем !important в верхней части, низ/бока оставляем как были. */
  .portfolio-tester-head {
    position: sticky;
    top: 0;
    z-index: 6;
    background: var(--bg-elevated);
    padding-top: calc(14px + env(safe-area-inset-top, 0px)) !important;
  }
  .portfolio-tester-head .h2 { font-size: 17px; }
  /* Низ модала: «Готово» во всю ширину, под палец. padding-bottom += safe-area-inset-bottom,
     чтобы кнопка не попала под индикатор-home/жест-бар у самого низа листа (таб-бар при
     открытом .scrim уже скрыт правилом ниже). Перебиваем inline padding (12px 18px). */
  .portfolio-tester-foot {
    position: sticky;
    bottom: 0;
    background: var(--bg-elevated);
    padding-bottom: calc(12px + env(safe-area-inset-bottom, 0px)) !important;
  }
  .portfolio-tester-foot .btn.primary { flex: 1; justify-content: center; min-height: 44px; }

  /* Сетка выбора (группы/бумаги) уже 1 колонка (см. общий блок выше). Паддинг сетки ужать. */
  .portfolio-tester-grid { padding: 12px; gap: 14px; }

  /* Блоки внутри модала держим в ширину экрана (без распирания таблицами/прогрессом). */
  .portfolio-tester-modal .block,
  .portfolio-tester-modal .block-body { min-width: 0; max-width: 100%; }

  /* 3 KPI выбора групп — стопкой (полные подписи не влезают 3-в-ряд на 360px). */
  .portfolio-run-summary { grid-template-columns: 1fr; }
  .portfolio-run-summary .kpi-cell { min-height: 0; padding: 10px 12px; }

  /* Список групп/бумаг → карточки .bcard со switch (эталон mobile стр.87-89, 97-99). */
  .portfolio-group-list,
  .portfolio-bond-list {
    max-height: none;
    overflow: visible;
    display: flex;
    flex-direction: column;
    gap: 8px;
    padding: 10px 12px;
  }
  .portfolio-group-row,
  .portfolio-bond-row {
    border: 1px solid var(--border-default);
    border-radius: 12px;
    background: var(--bg-container);
    padding: 10px 12px;
    gap: 8px;
  }
  .portfolio-group-row:last-child,
  .portfolio-bond-row:last-child { border-bottom: 1px solid var(--border-default); }
  .portfolio-group-row.active { border-color: var(--border-brand); }
  /* Карточка бумаги: тикер+имя слева (растягиваются, имя ellipsis) + switch справа в первой
     строке; меты группа·лот·ТФ·канал переносятся отдельной строкой ниже, инлайн через «·».
     6-колоночная десктоп-сетка → flex-wrap в 2 ряда. */
  .portfolio-bond-row {
    display: flex;
    flex-wrap: wrap;
    column-gap: 8px;
    row-gap: 5px;
    align-items: center;
  }
  /* .col занимает почти всю 1-ю строку (basis = 100% − место под switch), поэтому switch
     остаётся справа в 1-й строке, а все 4 меты целиком переносятся во 2-ю строку и идут
     инлайн через «·». Так group-мета не «прилипает» к 1-й строке между .col и switch. */
  .portfolio-bond-row > .col { order: 1; flex: 1 1 calc(100% - 44px); min-width: 0; }
  .portfolio-bond-row > .switch { order: 2; flex: 0 0 auto; }
  .portfolio-bond-meta {
    order: 3;
    flex: 0 0 auto;
    font-size: 10px;
    overflow: visible;
    white-space: nowrap;
  }
  /* разделитель-точки между метами на мобайле (на десктопе их роль играют колонки сетки) */
  .portfolio-bond-meta + .portfolio-bond-meta::before {
    content: "· ";
    color: var(--text-tertiary);
  }

  /* Подвал блока «Бумаги для прогона» (счётчик + прогресс + кнопка «Прогнать портфель»):
     на узких экранах три элемента в один ряд (space-between, nowrap) накладывались друг на
     друга и вылезали за край. Переносим в столбик: прогресс и кнопка — во всю ширину.
     :has — как в других местах этого файла. Десктоп (≥641) без изменений. */
  .portfolio-tester-modal .block-body > .row:has(.portfolio-run-progress) {
    flex-wrap: wrap;
    gap: 8px;
  }
  .portfolio-tester-modal .block-body > .row:has(.portfolio-run-progress) > .btn {
    flex: 1 1 100%;
    justify-content: center;
    min-height: 44px;
  }

  /* «Итог прогона» → «Итоги портфеля»: 4 KPI = 2 в ряд. */
  .portfolio-final-kpis { grid-template-columns: 1fr 1fr; }
  .portfolio-final-kpi {
    min-height: 0;
    padding: 11px 12px;
    gap: 4px;
    border-right: 0;
    border-bottom: 1px solid var(--border-default);
  }
  .portfolio-final-kpi:nth-child(odd) { border-right: 1px solid var(--border-default); }
  .portfolio-final-kpi:nth-last-child(-n + 2) { border-bottom: 0; }
  .portfolio-final-kpi b { font-size: 17px; }
  .portfolio-final-kpi .label, .portfolio-final-kpi small { font-size: 10px; }

  /* Шапка отчёта: на мобайле дублирующий h3 «Итоги портфеля» прячем (статус-eyebrow его уже
     несёт — эталон mobile: только eyebrow + строка-мета). Голову облегчаем, переносим. */
  .portfolio-final-head {
    flex-direction: column;
    align-items: flex-start;
    gap: 4px;
    padding: 10px 14px;
    background: transparent;
  }
  .portfolio-final-head h3 { display: none; }
  .portfolio-final-head .mono.muted { font-size: 10px; }
  .portfolio-final-status { padding: 12px 14px 4px; }

  /* 7 секций «Итоги портфеля» → стопкой в один столбец (эталон mobile: один .bcard). */
  .portfolio-final-grid { grid-template-columns: 1fr; }
  .portfolio-final-col {
    min-height: 0;
    gap: 14px;
    padding: 14px;
    border-right: 0;
    border-bottom: 1px solid var(--border-default);
  }
  .portfolio-final-col:last-child { border-bottom: 0; }
  .portfolio-final-line, .portfolio-final-line b { font-size: 13px; }
  /* highlight-строки тянутся в край колонки — синхронизируем с мобильным паддингом (14px). */
  .portfolio-final-line.highlight { margin: 0 -14px; padding: 8px 14px; }

  /* Низ отчёта: кнопки «по месяцам»/«сделки» во всю ширину, примечание мельче. */
  .portfolio-final-footer {
    flex-direction: column;
    align-items: stretch;
    gap: 10px;
    padding: 14px;
  }
  .portfolio-final-footer .row.gap-8 { display: flex; gap: 8px; }
  .portfolio-final-footer .row.gap-8 .btn { flex: 1; justify-content: center; min-height: 40px; }
  .portfolio-final-footer .muted { font-size: 10px; }

  /* Вкладки отчёта → mchip-чипы (горизонт-скролл), эталон mobile стр.125. */
  .portfolio-report-tabs {
    gap: 7px;
    padding: 10px 12px !important;
    border-bottom: 0;
    overflow-x: auto;
    -webkit-overflow-scrolling: touch;
    scrollbar-width: none;
    flex-wrap: nowrap;
  }
  .portfolio-report-tabs::-webkit-scrollbar { display: none; }
  .portfolio-report-tabs .tab {
    flex: 0 0 auto;
    margin-bottom: 0;
    padding: 7px 13px;
    border: 1px solid var(--border-strong);
    border-radius: 999px;
    background: var(--bg-neutral);
    color: var(--text-secondary);
    font-size: 12.5px;
    font-weight: 600;
    white-space: nowrap;
  }
  .portfolio-report-tabs .tab.active {
    background: var(--bg-brand);
    border-color: var(--bg-brand);
    color: var(--text-on-brand);
  }

  /* Таблицы отчёта: горизонт-скролл внутри обёртки (min-width задан inline на каждой таблице),
     высоту не ограничиваем, подсказку показываем. */
  .portfolio-report-wrap {
    max-height: none;
    overflow-x: auto;
    -webkit-overflow-scrolling: touch;
    max-width: 100%;
  }
  .portfolio-report-hint { display: block; font-size: 10px; padding: 8px 14px 0; }
}

/* ===================================================================
   ЭКРАН ВХОДА (логин). Сплит-экран на десктопе (бренд+смысл слева, вход
   справа), одностолбцовая центрированная вертикаль на мобайле (≤640px).
   Порт эталонов handoff/login/desktop.dc.html + mobile.dc.html на наши
   токены. Показывается, когда authState.authenticated === false.
   =================================================================== */
.login {
  display: grid;
  grid-template-columns: 1.1fr 0.9fr;
  min-height: 100vh;
  color: var(--text-primary);
  /* Пре-авторизация рендерится прямо в #root под body{overflow:hidden} — т.е. сам экран входа
     должен быть скролл-контейнером, иначе при длинной левой колонке (тезисы) низ обрезался и был
     недостижим. Даём .login определённую высоту вьюпорта + собственный вертикальный скролл; когда
     контент влезает — скролла нет; когда выше вьюпорта — страница прокручивается. На мобайле
     (≤640px) ничего не ломает: левая панель скрыта, форм-панель сохраняет min-height и теперь
     тоже скроллится через этот же контейнер. */
  height: 100vh;
  height: 100dvh;
  overflow-y: auto;
  /* Плавная инерционная прокрутка на iOS Safari (старые версии без неё «не докручивают»). */
  -webkit-overflow-scrolling: touch;
}
.login-pane-brand {
  position: relative;
  overflow: visible;
  padding: 48px 56px;
  display: flex;
  flex-direction: column;
  /* Центрируем смысловой блок по вертикали, когда он влезает. Скролл живёт на .login (грид-контейнер),
     поэтому при переполнении строка грида растёт ровно по контенту — лишнего свободного места нет,
     и justify-content:center не обрезает верх (прокрутка идёт сверху). */
  justify-content: center;
  /* …и по горизонтали: блок контента центрируется как единое целое. Все прямые дети
     (бренд-шапка, питч, дисклеймер) имеют одинаковую ширину (см. ниже) и выровнены
     влево внутри неё — поэтому у них единый левый край, а сам блок стоит по центру. */
  align-items: center;
  gap: 28px;
  background:
    radial-gradient(120% 90% at 15% 0%, rgba(31, 138, 91, 0.16), transparent 50%),
    var(--color-neutral-950);
  border-right: 1px solid var(--border-default);
}
/* Единая ширина прямых детей бренд-панели = ширине питча. Даёт общий левый край
   (логотип, заголовок, лид, карточка, тезисы, дисклеймер начинаются на одной линии),
   при этом весь блок остаётся отцентрованным по горизонтали (align-items:center выше). */
.login-pane-brand > .login-brandhead,
.login-pane-brand > .login-disclaimer-desktop {
  width: 460px;
  max-width: 100%;
}
.login-pane-form {
  display: flex;
  align-items: center;
  justify-content: center;
  padding: 48px;
  background: var(--bg-app);
}
.login-formcard {
  width: 380px;
  max-width: 100%;
  display: flex;
  flex-direction: column;
  gap: 22px;
}

/* Бренд-шапка левой панели */
.login-brandhead {
  display: flex;
  align-items: center;
  gap: 10px;
}
.login-brandhead-text { gap: 1px; }
.login-brandname {
  font-weight: 800;
  font-size: 18px;
  letter-spacing: -0.01em;
}
.login-brandsub {
  font-size: 10px;
  letter-spacing: 0.16em;
  color: var(--text-tertiary);
}

/* Питч (заголовок + лид + кривая + тезисы) */
.login-pitch {
  gap: 26px;
  /* Контент левой панели читается слева (единый левый край у заголовка, лида,
     карточки и тезисов). По горизонтали блок центрируется как единое целое —
     все прямые дети .login-pane-brand имеют одну ширину и выровнены влево. */
  width: 460px;
  max-width: 100%;
  align-items: flex-start;
  text-align: left;
}
.login-pitch-head { gap: 14px; }
.login-h1 {
  margin: 0;
  font-size: 38px;
  line-height: 1.15;
  font-weight: 700;
  letter-spacing: -0.02em;
}
.login-lead {
  margin: 0;
  font-size: 15px;
  line-height: 1.6;
  color: var(--text-secondary);
}

/* Карточка кривой капитала */
.login-curvecard {
  border: 1px solid var(--border-default);
  border-radius: 14px;
  background: var(--bg-container);
  padding: 16px 18px;
}
.login-curvecard-head {
  justify-content: space-between;
  align-items: baseline;
  margin-bottom: 8px;
}
.login-curvecard-label { font-size: 9px; }
.login-curvecard-period {
  font-size: 11px;
  color: var(--text-tertiary);
}

/* Тезисы */
.login-tenets { gap: 14px; }
.login-tenet {
  display: flex;
  gap: 11px;
  align-items: flex-start;
}
.login-tenet-ic {
  flex: 0 0 auto;
  width: 30px;
  height: 30px;
  border-radius: 8px;
  background: var(--bg-brand-tint);
  color: var(--text-brand);
  display: grid;
  place-items: center;
}
.login-tenet-text { gap: 1px; }
.login-tenet-title {
  font-size: 13px;
  font-weight: 600;
}
.login-tenet-sub { font-size: 12px; }

.login-disclaimer {
  font-size: 11px;
  color: var(--text-tertiary);
  line-height: 1.5;
}

/* Заголовок блока входа */
.login-formhead { gap: 6px; }
.login-h2 {
  margin: 0;
  font-size: 24px;
  font-weight: 700;
}
.login-formhead-sub { font-size: 13px; }

/* Блок кнопок входа */
.login-auth {
  display: flex;
  flex-direction: column;
  gap: 10px;
}
.login-oauth {
  width: 100%;
  display: flex;
  align-items: center;
  justify-content: center;
  gap: 10px;
  padding: 14px 18px;
  border-radius: 11px;
  font-family: inherit;
  font-size: 15px;
  font-weight: 600;
  cursor: pointer;
  border: 1px solid var(--border-strong);
  background: var(--bg-container);
  color: var(--text-primary);
  box-sizing: border-box;
  transition: border-color .12s, background .12s, filter .12s;
}
.login-oauth.primary {
  background: var(--bg-brand);
  border-color: var(--bg-brand);
  color: var(--text-on-brand);
}
.login-oauth.primary:hover:not(:disabled) { filter: brightness(1.05); }
.login-oauth.primary:disabled {
  cursor: default;
  filter: grayscale(0.35);
  opacity: 0.65;
}
.login-oauth.soon {
  color: var(--text-tertiary);
  cursor: default;
}
.login-soon-tag {
  margin-left: 4px;
  padding: 0 7px;
}
.login-ya {
  width: 22px;
  height: 22px;
  border-radius: 6px;
  background: #fc3f1d;
  color: #fff;
  display: grid;
  place-items: center;
  font-weight: 700;
  font-size: 15px;
  flex: 0 0 auto;
  font-family: var(--font-sans);
}
.login-divline {
  display: flex;
  align-items: center;
  gap: 12px;
  color: var(--text-tertiary);
  font-size: 11px;
}
.login-divline::before,
.login-divline::after {
  content: "";
  height: 1px;
  background: var(--border-default);
  flex: 1;
}

/* Промокод (опционально, [NEW]) */
.login-promo {
  display: flex;
  flex-direction: column;
  gap: 8px;
}
.login-promo-link {
  font-size: 12px;
  color: var(--text-tertiary);
  cursor: pointer;
  text-align: center;
  background: none;
  border: 0;
  padding: 2px;
  font-family: inherit;
}
.login-promo-link:hover { color: var(--text-secondary); }
.login-promo-field {
  display: flex;
  flex-direction: column;
  gap: 6px;
}
.login-promo-hint {
  font-size: 11px;
  line-height: 1.5;
}

.login-terms {
  font-size: 11px;
  line-height: 1.5;
  text-align: center;
}

/* Мобильная бренд-шапка: на десктопе скрыта, нижний дисклеймер тоже только мобайл. */
.login-mobile-brand { display: none; }
.login-disclaimer-mobile { display: none; }

@media (max-width: 640px) {
  /* Один столбец: левую смысловую панель прячем, показываем компактную мобильную шапку. */
  .login { grid-template-columns: 1fr; }
  .login-pane-brand { display: none; }
  .login-pane-form {
    align-items: flex-start;
    padding: 24px 20px 28px;
    background:
      radial-gradient(120% 50% at 50% 0%, rgba(31, 138, 91, 0.16), transparent 55%),
      var(--bg-app);
    min-height: 100vh;
    box-sizing: border-box;
  }
  .login-formcard {
    width: 100%;
    gap: 18px;
  }

  .login-mobile-brand {
    display: flex;
    flex-direction: column;
    align-items: center;
    text-align: center;
    gap: 14px;
    margin-top: 6px;
  }
  .login-brandname-mobile { font-size: 20px; }
  .login-h1-mobile {
    margin: 2px 0 0;
    font-size: 23px;
    line-height: 1.22;
    font-weight: 700;
    letter-spacing: -0.01em;
  }
  .login-lead-mobile {
    margin: 0;
    font-size: 13.5px;
    line-height: 1.55;
    color: var(--text-secondary);
  }
  .login-mobile-brand .login-h1-mobile,
  .login-mobile-brand .login-lead-mobile { max-width: 320px; }
  .login-mobile-brand svg[viewBox="0 0 460 100"] { margin: 6px 0 2px; }
  .login-mobile-chips {
    justify-content: center;
    gap: 8px;
    flex-wrap: wrap;
  }

  /* На мобайле блок входа центрируем по заголовку, кнопки — тач-таргет ≥50px. */
  .login-h2 { font-size: 22px; }
  .login-oauth {
    padding: 14px 16px;
    border-radius: 12px;
    min-height: 50px;
  }
  /* На десктопе нижний дисклеймер живёт в левой панели; на мобайле — под формой. */
  .login-disclaimer-mobile {
    display: block;
    font-size: 10.5px;
    text-align: center;
    margin-top: 4px;
  }
}

/* Profile */
.profile-screen {
  display: grid;
  gap: 16px;
  padding: 24px;
}
.profile-head {
  display: flex;
  align-items: flex-end;
  justify-content: space-between;
  gap: 16px;
}
.profile-avatar-large {
  width: 56px;
  height: 56px;
  border-radius: 999px;
  display: grid;
  place-items: center;
  background: var(--bg-brand);
  color: var(--text-on-brand);
  font-size: 18px;
  font-weight: 800;
}
.profile-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  gap: 16px;
}
/* Профиль: 9 блоков раскладываются в 2 колонки masonry (CSS multi-column), как в эталоне
   (handoff/screens desktop: columns:2; column-gap:16px). Каждый .block неразрывен и держит
   нижний отступ вместо grid-gap. На мобайле (≤640px) — 1 колонка (см. медиазапрос ниже). */
.profile-blocks {
  columns: 2;
  column-gap: 16px;
  /* грид-айтем профиля: без min-width:0 multi-column контейнер не сжимается до трека
     и распирает экран на мобайле (input width:100% даёт интринзик-ширину). */
  min-width: 0;
}
.profile-blocks > .block {
  break-inside: avoid;
  -webkit-column-break-inside: avoid;
  margin-bottom: 16px;
}
.profile-form {
  display: grid;
  gap: 14px;
  padding: 16px;
}
.profile-summary {
  display: grid;
  grid-template-columns: repeat(3, minmax(0, 1fr));
}
.profile-provider-list {
  display: grid;
}
.profile-provider-row,
.profile-toggle-row,
.profile-session {
  display: flex;
  align-items: center;
  gap: 12px;
  padding: 14px 16px;
  border-bottom: 1px solid var(--border-default);
  /* flex-строка не должна распирать блок своей интринзик-шириной (длинный subtitle +
     кнопка). Внутренняя текст-колонка уже min-width:0 и эллипсит. Иначе на мобайле
     эти строки вылезали за карточку (горизонт-скролл .scroll). */
  min-width: 0;
}
.profile-provider-row:last-child,
.profile-toggle-row:last-child {
  border-bottom: 0;
}
/* Текст-колонка тоггл-строки должна растягиваться, чтобы Switch прижимался к правому
   краю — тогда все переключатели в блоке стоят по одной линейке (как в макете, где
   строка = justify-content:space-between). Scoped к .profile-toggle-row, поэтому
   provider/session-строки (общее правило выше) не затронуты. min-width:0 — чтобы длинный
   subtitle эллипсился и не распирал строку на мобайле. */
.profile-toggle-row > .col {
  flex: 1 1 auto;
  min-width: 0;
}
.profile-provider-mark {
  width: 32px;
  height: 32px;
  border-radius: 7px;
  display: grid;
  place-items: center;
  background: var(--bg-neutral);
  color: var(--text-primary);
  border: 1px solid var(--border-default);
  font-weight: 800;
}
/* Текущая сессия: левая текст-колонка (бесклассовый <div> с email/способом входа) НЕ имела
   min-width:0 — длинный Yandex-email (без пробелов, не переносится) распирал её интринзик-шириной
   и выталкивал ряд «активна»+«Выйти» за правый край карточки (на мобайле ≤640px кнопка «Выйти»
   вылезала из блока, на десктопе — горизонт-скролл). Лечим: колонка сжимается (min-width:0),
   email/подпись эллипсятся (nowrap+ellipsis), правый ряд с кнопкой не сжимается (flex-shrink:0).
   Универсально безопасно: на широкой десктоп-карточке эллипсис не срабатывает. */
.profile-session > div:first-child {
  min-width: 0;
  flex: 1 1 auto;
}
.profile-session > div:first-child > .pri,
.profile-session > div:first-child > .muted {
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.profile-session > .row {
  flex-shrink: 0;
}
/* Способы входа: подпись провайдера (.sec/.muted — например email подключённого Yandex) тоже
   длинный токен без пробелов. Текст-колонка уже min-width:0 (инлайн), но без nowrap+ellipsis
   глифы вылезали за бокс и давали горизонт-скролл страницы. Эллипсим подпись. */
.profile-provider-row > .col > .sec,
.profile-provider-row > .col > .muted {
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.profile-save-state {
  display: flex;
  justify-content: flex-end;
}

/* ===================================================================
   МОБИЛЬНЫЙ КАРКАС (Фаза 1): нижний таб-бар + адаптация шапки + тач/сетки.
   Брейкпоинт ≤640px. На десктопе (≥641px) ничего не меняется.
   =================================================================== */
.mobile-tabbar { display: none; }
.mobile-more-scrim { display: none; }

@media (max-width: 640px) {
  /* Десктопный сайдбар прячем (у aside inline display:flex — нужен !important), контент на всю ширину */
  .shell > aside { display: none !important; }
  /* таб-бар фикс. снизу = 58px таба + safe-area; контент должен заканчиваться ВЫШЕ него,
     иначе последняя строка/пагинация прячутся под панелью (нужен запас + safe-area) */
  .scroll { padding-bottom: calc(74px + env(safe-area-inset-bottom, 0px)); }

  /* Нижний таб-бар */
  .mobile-tabbar {
    display: flex;
    position: fixed; left: 0; right: 0; bottom: 0; z-index: 90;
    background: var(--color-neutral-950);
    border-top: 1px solid var(--border-default);
    padding-bottom: env(safe-area-inset-bottom, 0px);
  }
  .mobile-tab {
    flex: 1; min-height: 58px; padding: 6px 0;
    display: flex; flex-direction: column; align-items: center; justify-content: center; gap: 3px;
    border: 0; background: none; cursor: pointer; font-family: inherit;
    color: var(--text-tertiary);
  }
  .mobile-tab span { font-size: 10px; line-height: 1; }
  .mobile-tab.active { color: var(--text-brand); }

  /* «Ещё» — боттом-шит */
  .mobile-more-scrim {
    display: block; position: fixed; inset: 0; z-index: 95;
    background: var(--bg-overlay); backdrop-filter: blur(2px);
  }
  .mobile-more-sheet {
    position: fixed; left: 0; right: 0; bottom: 0; z-index: 96;
    background: var(--bg-elevated); border-top: 1px solid var(--border-default);
    border-radius: 14px 14px 0 0;
    padding: 6px 10px calc(14px + env(safe-area-inset-bottom, 0px));
    display: flex; flex-direction: column; gap: 2px;
  }
  .mobile-more-handle { width: 40px; height: 4px; border-radius: 2px; background: var(--border-strong); margin: 6px auto 10px; }
  .mobile-more-robot {
    display: flex; align-items: center; gap: 10px; font-size: 13px;
    padding: 8px 12px 12px; margin-bottom: 6px; border-bottom: 1px solid var(--border-default);
  }
  .mobile-more-item {
    display: flex; align-items: center; gap: 12px; min-height: 50px;
    padding: 0 12px; border: 0; background: none; cursor: pointer; border-radius: 8px;
    color: var(--text-primary); font-family: inherit; font-size: 14px;
  }
  .mobile-more-item.active { background: var(--bg-brand-tint); color: var(--text-brand); }
  .mobile-more-item .count {
    font-family: var(--font-mono); font-size: 11px; padding: 1px 7px; border-radius: 999px;
    background: var(--bg-neutral); color: var(--text-tertiary);
  }

  /* Тач-таргеты */
  .btn { min-height: 44px; }
  .btn.sm { min-height: 40px; }

  /* KPI — 2 в ряд */
  .kpi-grid { grid-template-columns: 1fr 1fr; }

  /* Модалки → боттом-шиты */
  .scrim { place-items: end stretch; }
  .modal {
    min-width: 0; width: 100%; max-width: 100%;
    border-radius: 14px 14px 0 0; max-height: 88vh;
  }

  /* Шапка на мобилке: убираем хлебные крошки и аватар, показываем номер счёта; из статов
     оставляем Баланс + Своб. (разделители и кредит прячем). Inline gap/padding шапки бьём
     !important, иначе номер счёта + два значения не помещаются. */
  .topbar { gap: 8px !important; padding: 0 12px !important; }
  .crumb { display: none; }
  .topbar-avatar { display: none; }
  .topbar-account { display: inline-flex; align-items: center; color: var(--text-secondary); font-size: 12px; white-space: nowrap; }
  .topbar-stats { font-size: 12px; }
  .topbar-stats .divider-v { display: none; }
  .topbar-stats > span:nth-child(n+4) { display: none; }

  /* AccountSwitcher на мобайле: поповер → боттом-шит над таб-баром, перехватчик затемняет фон. */
  .acct-switch-scrim {
    z-index: 110;
    background: var(--bg-overlay); backdrop-filter: blur(2px);
  }
  .acct-switch-topbar .acct-switch-menu,
  .acct-switch-menu {
    position: fixed; z-index: 111;
    left: 0; right: 0; top: auto;
    bottom: calc(58px + env(safe-area-inset-bottom, 0px));
    min-width: 0;
    border-radius: 14px 14px 0 0;
    box-shadow: 0 -10px 30px rgba(0, 0, 0, .45);
  }
  .acct-switch-menu-handle {
    display: block; width: 40px; height: 4px; border-radius: 2px;
    background: var(--border-strong); margin: 2px auto 10px;
  }
  .acct-switch-menu-list { max-height: 52vh; }
  .acct-switch-item { min-height: 52px; }
  .acct-switch-manage { min-height: 50px; }
}

/* ============================================================
   ОБЗОР СЧЁТА — rev.4 дашборд (handoff/overview/desktop.dc.html).
   Классы портированы из эталона на НАШИ токены. Данные — реальные
   (см. screen-overview.jsx), числа из макета НЕ используются.
   ============================================================ */

/* Сетки rev.4 */
.ovr4-grid2 { display: grid; grid-template-columns: 1.15fr 0.85fr; gap: 16px; align-items: start; }
.ovr4-grid3 { display: grid; grid-template-columns: repeat(3, 1fr); gap: 14px; }
.ovr4-grid-money { display: grid; grid-template-columns: 1fr 1fr; gap: 12px; }

/* Базовая карточка дашборда (= .card эталона) */
.ovr4-card {
  background: var(--bg-container);
  border: 1px solid var(--border-default);
  border-radius: 14px;
  padding: 16px;
}

/* Статус-полоса (лицензия/сервер/робот) */
.ovr4-status {
  display: flex; align-items: center; gap: 16px; flex-wrap: wrap;
  padding: 12px 16px;
}

/* Денежная карточка */
.ovr4-moneycard {
  background: var(--bg-container);
  border: 1px solid var(--border-default);
  border-radius: 14px;
  padding: 14px 16px;
  display: flex; flex-direction: column; gap: 6px;
  position: relative; overflow: hidden;
}
.ovr4-money-label { font-size: 9px; font-family: var(--font-mono); text-transform: uppercase; letter-spacing: .04em; color: var(--text-tertiary); }
.ovr4-money-big { font-family: var(--font-mono); font-size: 20px; font-weight: 700; }
.ovr4-money-big.gold { color: var(--text-money); }
.ovr4-money-sub { font-family: var(--font-mono); font-size: 10px; color: var(--text-tertiary); }

/* Витрина-HERO «Прибыль за сегодня» */
.ovr4-vitrina {
  background: radial-gradient(120% 120% at 80% 0%, oklch(from var(--color-success-500) l c h / 0.18), transparent 55%), var(--color-neutral-950);
  border: 1px solid var(--border-strong);
  border-radius: 18px;
  padding: 22px;
  position: relative; overflow: hidden;
}
.ovr4-vitrina-eyebrow { font-family: var(--font-mono); font-size: 10px; letter-spacing: .14em; color: var(--text-tertiary); }
.ovr4-vitrina-big { font-family: var(--font-mono); font-size: 40px; font-weight: 800; color: var(--text-money); letter-spacing: -0.02em; }
.ovr4-vitrina-foot { display: flex; gap: 0; margin-top: 12px; border-top: 1px solid var(--border-default); padding-top: 12px; }
.ovr4-vitrina-foot .col { gap: 1px; flex: 1; }
.ovr4-vitrina-period { font-family: var(--font-mono); font-size: 16px; font-weight: 700; }

/* Точечная легенда (линии графика / донат) */
.ovr4-legend-dot { width: 9px; height: 9px; border-radius: 50%; display: inline-block; flex-shrink: 0; }

/* Маленький сегмент-переключатель периода графика */
.ovr4-seg { display: inline-flex; background: var(--bg-neutral); border: 1px solid var(--border-default); border-radius: 8px; padding: 2px; }
.ovr4-seg button { border: 0; background: transparent; color: var(--text-secondary); padding: 5px 10px; font-size: 11px; font-weight: 600; border-radius: 6px; font-family: inherit; cursor: pointer; }
.ovr4-seg button.on { background: var(--bg-container); color: var(--text-primary); }

/* Компактные строки P&L (НКД / Фикс) */
.ovr4-pnl-row { display: grid; grid-template-columns: 1.4fr 1fr 1fr; gap: 8px; padding: 8px 0; border-bottom: 1px solid var(--border-default); align-items: baseline; }
.ovr4-pnl-row:last-child { border-bottom: 0; }
.ovr4-pnl-row .lbl { font-size: 12px; color: var(--text-secondary); }
.ovr4-pnl-row .v { font-family: var(--font-mono); font-size: 13px; font-weight: 600; text-align: right; }
.ovr4-pnl-head { font-family: var(--font-mono); font-size: 9px; text-transform: uppercase; letter-spacing: .05em; color: var(--text-tertiary); text-align: right; }

/* График: обёртка для hover-кросхэйра + тултипа */
.ovr4-chart-wrap { position: relative; }
.ovr4-chart-cross { position: absolute; top: 0; bottom: 0; width: 1px; border-left: 1px dashed var(--text-tertiary); pointer-events: none; }
.ovr4-chart-tip {
  position: absolute; top: 8px;
  background: var(--bg-elevated); border: 1px solid var(--border-strong); border-radius: 8px;
  padding: 8px 10px; box-shadow: 0 8px 22px oklch(0 0 0 / 0.45);
  pointer-events: none; white-space: nowrap; z-index: 5;
}

/* Донат-распределение */
.ovr4-donut-legend { display: flex; flex-direction: column; gap: 7px; flex: 1; min-width: 0; }
.ovr4-donut-legend .row { justify-content: space-between; }

/* Ряд «ближайшие купоны/погашения» */
.ovr4-due-row { display: flex; justify-content: space-between; align-items: center; padding: 7px 0; border-bottom: 1px solid var(--border-default); gap: 12px; }
.ovr4-due-row:last-child { border-bottom: 0; }

/* ===== «Движение по счёту»: раскрываемая строка (тап → полный текст + детали) ===== */
.ovr-move-row { padding: 10px 16px; cursor: pointer; transition: background .15s ease; outline: none; }
.ovr-move-row:hover { background: var(--bg-neutral); }
.ovr-move-row:focus-visible { background: var(--bg-neutral); box-shadow: inset 0 0 0 1px var(--border-strong); }
.ovr-move-row.open { background: var(--bg-neutral); }
/* Заголовок операции: по умолчанию одна строка с многоточием, в раскрытом виде — целиком (перенос). */
.ovr-move-title { font-size: 13px; min-width: 0; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.ovr-move-title.open { overflow: visible; white-space: normal; word-break: break-word; line-height: 1.35; }
.ovr-move-sub { overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.ovr-move-row.open .ovr-move-sub { white-space: normal; word-break: break-word; }
/* Шеврон-индикатор: вниз в раскрытом состоянии */
.ovr-move-chev { display: inline-flex; color: var(--text-tertiary); transition: transform .18s ease; }
.ovr-move-chev.open { transform: rotate(90deg); }
/* Блок деталей под строкой */
.ovr-move-detail {
  margin-top: 8px; padding-top: 8px; border-top: 1px dashed var(--border-default);
  display: flex; flex-direction: column; gap: 5px;
  animation: ovr-move-reveal .18s ease-out;
}
.ovr-move-detail-row { display: flex; justify-content: space-between; align-items: baseline; gap: 12px; }
.ovr-move-detail-row .label { flex-shrink: 0; }
.ovr-move-detail-row > span:last-child { font-size: 12px; color: var(--text-secondary); text-align: right; word-break: break-word; min-width: 0; }
@keyframes ovr-move-reveal { from { opacity: 0; transform: translateY(-2px); } to { opacity: 1; transform: none; } }

/* ===== МОБИЛЬНЫЙ «ОБЗОР» (Фаза 2): стек 2-кол сеток + чипсы-статусы + компактнее ===== */
.ovr-chips { display: none; }
@media (max-width: 640px) {
  .ovr-screen { padding: 12px !important; gap: 12px !important; }
  /* Корень «Обзора» хостит .scrim+.gr-modal (Глобальные правила) как fixed-потомков. Входная
     .fadein несёт transform (translateY 4px), а любой transform у предка делает его containing
     block для fixed → и .scrim, и сам полноэкранный лист считались от .ovr-screen (со сдвигом
     на топбар ~52px), height:100dvh свешивался ниже фолда и срезал футер «Сохранить». Тот же
     кейс и фикс, что у .bonds-screen-root/.corr-screen/.accounts-screen: появление без сдвига
     (только opacity) — fixed-лист снова считается от вьюпорта, футер прибит к низу экрана.
     Десктоп (≥641) не трогаем — там модалка центрируется и обрезки нет. */
  .ovr-screen.fadein { animation: fadein-noshift .18s ease-out; }
  .ovr-head .h1 { display: none; }
  .ovr-2col { grid-template-columns: 1fr !important; gap: 12px !important; }
  .pnl-2col { grid-template-columns: 1fr !important; gap: 12px !important; }
  .ovr-chips {
    display: flex; gap: 6px; overflow-x: auto; scrollbar-width: none;
    -webkit-overflow-scrolling: touch;
  }
  .ovr-chips::-webkit-scrollbar { display: none; }
  .ovr-chip {
    flex-shrink: 0; font-size: 11px; white-space: nowrap;
    border: 1px solid var(--border-default); border-radius: 999px;
    padding: 5px 11px; color: var(--text-secondary);
  }

  /* rev.4: все 2/3-кол сетки дашборда → одна колонка, ужимаем витрину/карточки */
  .ovr4-grid2, .ovr4-grid3, .ovr4-grid-money { grid-template-columns: 1fr !important; gap: 12px !important; }
  .ovr4-vitrina { padding: 16px; border-radius: 16px; }
  .ovr4-vitrina-big { font-size: 30px; }
  .ovr4-vitrina-period { font-size: 13px; }
  .ovr4-card { padding: 13px; }
  .ovr4-moneycard { padding: 11px; }
  .ovr4-money-big { font-size: 16px; }
  /* статус-полоса аккуратно переносится, кнопка «Продлить сервер» остаётся достижимой */
  .ovr4-status { padding: 12px; row-gap: 10px; }
  .ovr4-status .divider-v { display: none; }
  .ovr4-status .spacer { display: none; }
  /* «Движение по счёту»: ужать паддинг строки; детали — лейбл над значением (без тесноты) */
  .ovr-move-row { padding: 10px 13px; }
  .ovr-move-detail-row { flex-direction: column; align-items: stretch; gap: 1px; }
  .ovr-move-detail-row > span:last-child { text-align: left; }
}

/* ===== МОБИЛЬНЫЕ КАРТОЧКИ ТАБЛИЦ (Фаза 3): table.t.t-cards → строки-карточки ===== */
@media (max-width: 640px) {
  .t-cards thead { display: none; }
  .t-cards, .t-cards tbody { display: block; }
  /* карточка = аккуратная 2-колоночная сетка; бейдж/тикер/действие — во всю ширину */
  .t-cards tbody tr {
    display: grid; grid-template-columns: 1fr 1fr; gap: 6px 16px; align-items: start;
    border: 1px solid var(--border-default); border-radius: 10px;
    margin: 0 0 10px; padding: 12px 13px; background: var(--bg-container); cursor: default;
  }
  .t-cards tbody tr:hover { background: var(--bg-container); }
  /* «Позиции»: левая кромка карточки = цвет источника (робот/резерв/коррекция) */
  .pos-screen .t-cards tbody tr { border-left: 3px solid var(--row-accent, var(--border-default)); }
  /* поле = «метка слева · значение справа», значения выстраиваются ровными колонками */
  .t-cards tbody td {
    display: flex; align-items: baseline; justify-content: space-between; gap: 8px;
    margin: 0; padding: 0; border: 0; font-size: 12px; min-width: 0;
    text-align: left; white-space: nowrap; overflow: hidden;
  }
  .t-cards tbody td::before {
    content: attr(data-label); color: var(--text-tertiary); font-size: 11px; flex-shrink: 0;
  }
  /* тикер — крупно, во всю ширину (имя может переноситься) */
  .t-cards tbody td.td-head {
    grid-column: 1 / -1; display: block; margin: 0 0 1px; padding: 0; font-size: 14px;
    white-space: normal; overflow: visible;
  }
  .t-cards tbody td.td-head::before { content: none; }
  /* статус-бейдж — во всю ширину сверху */
  .t-cards tbody td.td-badge { grid-column: 1 / -1; display: block; margin: 0; padding: 0; overflow: visible; }
  .t-cards tbody td.td-badge::before { content: none; }
  .t-cards tbody td.td-hide { display: none; }
  /* действие — кнопка во всю ширину снизу */
  .t-cards tbody td.td-action {
    grid-column: 1 / -1; display: block; margin: 5px 0 0; padding: 0; overflow: visible;
  }
  .t-cards tbody td.td-action::before { content: none; }
  .t-cards tbody td.td-action .btn { width: 100%; justify-content: center; min-height: 40px; }
  .t-cards tbody td[colspan] { grid-column: 1 / -1; display: block; text-align: center; overflow: visible; }
  .t-cards tbody td[colspan]::before { content: none; }
}

/* ===== МОБИЛЬНАЯ «ТОРГОВЛЯ»: график во всю ширину, стакан ПОД ним и поделён надвое ===== */
@media (max-width: 640px) {
  /* фикс. высота 325px клипала стакан под графиком (его не было видно) → авто, чтобы влезли оба */
  .trade-live-grid { grid-template-columns: 1fr; height: auto; }
  .chart-inline-stage .trade-live-grid { height: auto; }
  /* строка с графиком внутри ШИРОКОЙ таблицы (min-width 1080): прижимаем контент к левому
     краю окна и даём ширину экрана — иначе график уезжал в горизонтальный скролл таблицы */
  .chart-inline-reveal { position: sticky; left: 0; width: 100vw; max-width: 100vw; }

  /* стакан надвое: слева ПРОДАЖИ (asks), справа ПОКУПКИ (bids), цена сверху во всю ширину */
  .orderbook {
    display: grid; grid-template-columns: 1fr 1fr; column-gap: 10px; row-gap: 2px;
    border-left: 0; border-top: 1px solid var(--border-default); padding-bottom: 6px;
  }
  .orderbook-head { grid-column: 1 / -1; order: 1; }
  .orderbook-grid-head { display: none; }            /* «Лот Цена Лот» к 2 колонкам не подходит */
  .orderbook-mid { grid-column: 1 / -1; order: 2; margin: 2px 12px 4px; }
  .orderbook-side { order: 3; padding: 0 12px; }
  .orderbook-side.asks { grid-column: 1; }
  .orderbook-side.bids { grid-column: 2; }
  /* заголовки колонок без правки разметки */
  .orderbook-side::before {
    display: block; padding: 0 2px 3px; font-family: var(--font-mono); font-size: 9px;
    text-transform: uppercase; letter-spacing: .04em;
  }
  .orderbook-side.asks::before { content: "Продажи"; color: var(--text-error); }
  .orderbook-side.bids::before { content: "Покупки"; color: var(--text-success); }
  .orderbook-empty { grid-column: 1 / -1; order: 3; }
  /* в каждой половине — только 2 значащие ячейки (пустая треть схлопнута), строки выше под палец */
  .orderbook-row { min-height: 17px; font-size: 10px; }
  .orderbook-side.asks .orderbook-row { grid-template-columns: 1fr auto; }
  .orderbook-side.asks .orderbook-row > span:nth-child(1) { display: none; }
  .orderbook-side.bids .orderbook-row { grid-template-columns: auto 1fr; }
  .orderbook-side.bids .orderbook-row > span:nth-child(3) { display: none; }
}

/* ===== МОБИЛЬНАЯ «ТОРГОВЛЯ»: верхние строки (вкладки групп + фильтр/поиск) под ≤640px =====
   Десктоп-рестрактур сделал ROW A/B на «row wrap» + inline space-between. На телефоне
   space-between при переносе разносит контент уродливыми дырами, а 220px-инпут и 4-кнопочный
   сегмент не влезают в строку. Эти классы — единственная зацепка (см. screen-bonds.jsx);
   правила живут ТОЛЬКО здесь, десктоп (≥641px) не затрагивается. */
@media (max-width: 640px) {
  /* ROW A — строка вкладок групп: прижать всё влево (а не space-between по краям при переносе),
     дать строке переноситься; группы-вкладки уже переносятся (inline flexWrap). */
  .bonds-grouptabs-row { justify-content: flex-start !important; }

  /* Кнопки «Глобальные правила»/«Портфельный тестер» — на отдельную строку во всю ширину,
     каждая тянется и делит ширину пополам; повыше под палец (btn sm низкие). */
  .bonds-grouptabs-actions {
    flex-basis: 100%;
    width: 100%;
    align-self: stretch !important;
  }
  .bonds-grouptabs-actions .btn.sm {
    flex: 1 1 0;
    min-height: 40px;
    justify-content: center;
  }

  /* ROW B — строка фильтра: тоже влево, без распорки по краям. */
  .bonds-filter-row { justify-content: flex-start !important; }

  /* Сегмент-фильтр и поиск стекаются во всю ширину (раньше 4-кнопочный сегмент + 220px-инпут
     вылезали за экран). */
  .bonds-filter-controls { width: 100%; }
  /* Сегмент во всю ширину, 4 кнопки делят её поровну (min-height 40px уже задан .seg button выше). */
  .bonds-filter-controls .seg { width: 100%; }
  .bonds-filter-controls .seg button { flex: 1 1 0; }
  /* Инпут поиска: убить фиксированные inline 220px → во всю ширину строки. */
  .bonds-filter-controls .input {
    flex: 1 1 100%;
    width: 100% !important;
    min-width: 0;
  }

  /* Индикатор «сортировка: …» — низкий приоритет на телефоне; прячем, чтобы не теснил. */
  .bonds-filter-sort { display: none; }
}

/* ===== ПАЧКА ФИКСОВ ПО QA-АУДИТУ (сетки→1кол, тач-таргеты, паддинг, переносы) ===== */
@media (max-width: 640px) {
  /* экраны: компактный паддинг (был inline 24px → жал контент) */
  .col.gap-16.fadein { padding: 12px !important; }

  /* «Позиции»: держим ширину в экране (без уноса в чёрную область при горизонт. скролле)
     и стекаем шапку — заголовок сверху, кнопки/фильтр снизу во всю ширину (раньше длинная
     строка с 4-кнопочным фильтром вылезала за экран). */
  .pos-screen { overflow-x: clip; }
  .pos-head { flex-direction: column; align-items: stretch !important; gap: 10px; }
  .pos-head-ctl { flex-wrap: wrap; }
  .pos-head-ctl .seg { width: 100%; }

  /* 2-колоночные сетки → 1 колонка (профиль, настройки бумаги, тестер, группы) */
  .profile-grid,
  .settings-grid,
  .portfolio-tester-grid,
  .tester-top-grid,
  .grid-2 { grid-template-columns: 1fr !important; }
  /* Профиль (masonry) → 1 столбец на мобайле */
  .profile-blocks { columns: 1 !important; }
  /* ПУНКТ 3 — поля «Настроек группы» парами по 2 в ряд и на мобайле (эталон mobile.dc.html
     стр.129-130: Лимит+%, Сетки+Макс, Таймфрейм во всю ширину). Узкие числовые поля влезают
     в 2 колонки на 360px; перебиваем общий 1fr выше только для этой сетки. */
  .group-settings-panel .settings-grid { grid-template-columns: 1fr 1fr !important; column-gap: 10px; }
  /* minmax(0,1fr): без этого «1fr» = minmax(auto,1fr), и длинное значение Kpi («Не подключены»)
     распирало колонку за край. Ячейки и значение тоже учим сжиматься/переносить. */
  .profile-summary { grid-template-columns: minmax(0, 1fr) minmax(0, 1fr); }
  .profile-summary .kpi-cell { min-width: 0; }
  .profile-summary .kpi-value { min-width: 0; overflow-wrap: anywhere; }
  .portfolio-bond-row { grid-template-columns: minmax(0, 1fr) auto; }
  .portfolio-bond-toolbar { flex-wrap: wrap; }
  .portfolio-bond-toolbar .input { min-width: 0; }
  .portfolio-run-progress { min-width: 0 !important; width: 100%; }

  /* тач-таргеты под палец */
  .btn.icon { min-width: 44px; min-height: 44px; }
  .seg button { min-height: 40px; }
  .auto-pill { width: 36px; height: 36px; }
  .chart-tool { height: 36px; min-width: 36px; }
  .chart-layer-toggle { min-height: 40px; }

  /* длинные заголовки переносятся, а не вылезают за экран */
  .h1, .h2 { white-space: normal; word-break: break-word; }
  /* Шапка блока (.block-head): eyebrow и заголовок переносятся, а не упираются в правый край.
     .eyebrow глобально nowrap (см. выше) — на узких экранах это распирало шапку; здесь
     разрешаем перенос. Десктоп (≥641) не трогаем. */
  .block-head .eyebrow { white-space: normal; overflow-wrap: anywhere; }
  .block-head-title { white-space: normal; overflow-wrap: anywhere; }
}

/* мобайл «Настройки группы»: 1 колонка + вкладки Настройки/Фильтры/Корзины (десктоп не трогаем) */
.group-detail-mobtabs { display: none; }
/* статичный капшен «Настройки группы» — только мобайл; на десктопе скрыт (имя = крупный h1) */
.group-detail-caption { display: none; }
@media (max-width: 640px) {
  .group-detail-body { grid-template-columns: 1fr !important; }
  /* КОСЯК 1 — шапка столбиком (эталон mobile.dc.html стр.125): заголовок/имя/описание сверху,
     кнопки Пауза/Удалить — ОТДЕЛЬНЫМ рядом ПОД описанием, не справа от него (там row не
     переносился → кнопки жались справа и «болтались»). На десктопе ≥641 — без изменений. */
  .group-detail-head-row {
    flex-direction: column; align-items: stretch !important; gap: 12px;
  }
  .group-detail-head-row > .group-detail-head { width: 100%; }
  .group-detail-head-actions {
    width: 100%; flex-wrap: wrap; gap: 8px;
  }
  .group-detail-head-actions > .btn {
    flex: 1 1 0; min-width: 0; justify-content: center; min-height: 42px;
  }
  /* ПУНКТ 2 — иерархия шапки (эталон mobile.dc.html стр.124): капшен «Настройки группы»
     18px/700 сверху, имя группы ПОД ним мельче и muted. Имя остаётся редактируемым
     (карандаш/инлайн-инпут не трогаем — только размер/цвет). */
  .group-detail-caption {
    display: block; font-size: 18px; font-weight: 700; color: var(--text-primary);
    line-height: 1.2; margin-top: 2px;
  }
  .group-detail-head { gap: 1px !important; }
  /* имя-как-подзаголовок: было крупным h1 26px, теперь muted ~11px */
  .group-detail-head .group-title-edit { min-height: 0; gap: 4px; }
  .group-detail-head .group-title-edit .h1 {
    font-size: 11px; font-weight: 500; color: var(--text-tertiary);
    white-space: normal; word-break: break-word;
  }
  .group-detail-head .group-title-input {
    width: min(280px, 70vw); height: 28px; font-size: 13px; font-weight: 600;
  }
  /* карандаш-кнопку под мелкий подзаголовок ужимаем */
  .group-detail-head .group-title-edit .btn.icon { min-width: 28px; min-height: 28px; }
  /* описание группы под именем — ещё мельче, чтоб не спорило с капшеном */
  .group-detail-head > .muted { font-size: 11px !important; }
  .group-detail-mobtabs {
    display: flex; gap: 2px; margin-bottom: 12px;
    border-bottom: 1px solid var(--border-default);
  }
  .group-detail-mobtabs .gdm-tab {
    flex: 1 1 0; min-height: 42px; border: 0; background: none; cursor: pointer;
    font-family: inherit; font-size: 13px; font-weight: 600; color: var(--text-tertiary);
    border-bottom: 2px solid transparent;
  }
  .group-detail-mobtabs .gdm-tab.on { color: var(--text-brand); border-bottom-color: var(--border-brand); }
  /* показываем только активную секцию */
  .group-detail-body[data-mobile-section="settings"] .gs-filters,
  .group-detail-body[data-mobile-section="settings"] .gs-baskets,
  .group-detail-body[data-mobile-section="filters"] .gs-settings,
  .group-detail-body[data-mobile-section="filters"] .gs-baskets,
  .group-detail-body[data-mobile-section="baskets"] .gs-settings,
  .group-detail-body[data-mobile-section="baskets"] .gs-filters { display: none !important; }
  /* КОСЯК 2 — savebar НЕ липкий на мобайле (эталон mobile.dc.html стр.184: «savebar» внизу,
     контент над ним). Раньше был position:sticky;bottom:58px ВНУТРИ скролл-контента секции —
     при частичной прокрутке секции он всплывал ПОВЕРХ полей ниже («Тип группы» прятался под
     ним). Делаем static: ряд «сохранено/Сброс/Сохранить» просто завершает контент, поля над
     ним и не уезжают под него. Нижний отступ секции (.scroll padding-bottom:74px) уже уводит
     контент выше фиксированного таб-бара. Десктоп ≥641 — без изменений. */
  .group-detail-body .gs-savebar {
    position: static; z-index: auto;
    background: var(--bg-app);
    border-top: 1px solid var(--border-default);
    padding: 12px 0 calc(8px + env(safe-area-inset-bottom, 0px));
    margin-top: 10px;
  }
}

/* ===== Корзины бумаг (панель «Бумаги в группе»): мобильная адаптация по эталону
   handoff/trading/mobile.dc.html стр.143-178. Десктоп НЕ трогаем. ===== */
/* классы .mchip / .bcard в проекте отсутствовали — заводим 1:1 как в эталоне */
.mchip { flex: 0 0 auto; padding: 8px 14px; border-radius: 999px; border: 1px solid var(--border-strong); background: var(--bg-neutral); color: var(--text-secondary); font-size: 12.5px; font-weight: 600; display: inline-flex; align-items: center; gap: 7px; white-space: nowrap; cursor: pointer; }
.mchip.on { background: var(--bg-brand); border-color: var(--bg-brand); color: var(--text-on-brand); }
.bcard { background: var(--bg-container); border: 1px solid var(--border-default); border-radius: 14px; position: relative; overflow: hidden; }
.bcard::before { content: ""; position: absolute; left: 0; top: 0; bottom: 0; width: 4px; background: var(--accent, var(--border-strong)); z-index: 1; }
.gs-bcard { padding: 10px 12px; cursor: pointer; }
.gs-bcard.active { border-color: var(--border-brand); }
/* по умолчанию (десктоп) мобильные элементы спрятаны, таблицы видны как раньше */
.gs-mob-tabs, .gs-mob-cards, .gs-scroll-hint { display: none; }

/* ===== ЧС-модалка · мобильные карточки списка добавления (эталон mobile.dc.html стр.255-256).
   Десктоп-таблица (≥641) НЕ трогается. ===== */
.gs-bl-cards { display: none; }
@media (max-width: 640px) {
  /* инлайн width:940 на .modal перебивает дженерик-боттомшит → возвращаем во всю ширину */
  .gs-bl-modal { width: 100% !important; max-width: 100% !important; }
  /* таблица 5 колонок в боттом-шите тесная → прячем, показываем карточки */
  .gs-bl-modal .gs-bl-desk { display: none; }
  .gs-bl-cards { display: flex; flex-direction: column; gap: 8px; min-width: 0; }
  .gs-bl-card { padding: 10px 12px; }
  .gs-bl-card-head { display: flex; align-items: center; justify-content: space-between; gap: 10px; }
  .gs-bl-card-head .btn { flex: 0 0 auto; }
  .gs-bl-card-ticker { font-weight: 700; font-size: 13px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
  .gs-bl-card-sub { font-size: 10.5px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
  .gs-bl-card-ex { font-size: 10px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
  .gs-bl-empty {
    padding: 14px 12px; text-align: center; color: var(--text-tertiary);
    font-size: 12px; border: 1px dashed var(--border-default); border-radius: 10px;
  }
}

@media (max-width: 640px) {
  /* КЛЮЧЕВОЕ: grid/flex-элементы по умолчанию имеют min-width:auto и распирают трек
     контентом (широкая таблица → панель шире экрана → горизонт. скролл всей страницы).
     Фиксируем min-width:0 по всей цепочке до обёртки таблицы, чтобы 1fr-трек держал
     ширину экрана, а .gs-tbl-wrap скроллил таблицу ВНУТРИ себя. */
  .group-detail-body, .group-detail-body > .group-settings-panel { min-width: 0; }
  .gs-baskets, .gs-baskets > .block, .gs-baskets .block-body { min-width: 0; max-width: 100%; }
  .gs-baskets .gs-tbl-wrap, .gs-baskets .gs-mob-cards, .gs-baskets .gs-mob-tabs { min-width: 0; max-width: 100%; }
  /* десктопный ряд вкладок и десктопные таблицы белого/чёрного — прячем */
  .gs-baskets .gs-desk-tabs, .gs-baskets .gs-desk-only { display: none !important; }
  /* мобильный ряд чипов + кнопка «Из справочника» во всю ширину */
  .gs-mob-tabs { display: block; padding: 10px 14px 0; }
  .gs-mob-chips { display: flex; gap: 6px; overflow-x: auto; padding-bottom: 10px; -webkit-overflow-scrolling: touch; }
  .gs-mob-chips .mchip { padding: 6px 12px; }
  .gs-mob-picker { width: 100%; justify-content: center; margin-bottom: 10px; }
  /* широкие таблицы Кандидаты/Настроено — скроллятся вбок внутри обёртки, страница не вылезает */
  .gs-baskets .gs-tbl-wrap { overflow-x: auto; -webkit-overflow-scrolling: touch; }
  .gs-baskets .gs-tbl-wrap > table.t { min-width: 560px; }
  .gs-scroll-hint { display: block; padding: 6px 14px 0; }
  /* белый/чёрный — карточки */
  .gs-mob-cards { display: flex; flex-direction: column; gap: 8px; padding: 10px 14px; }
  /* превью-график выбранной бумаги закреплён сверху списка (эталон стр.146) */
  .gs-baskets .group-candidate-chart {
    position: sticky; top: 0; z-index: 6;
    background: var(--bg-app);
  }
  /* Шапка превью-графика: имя бумаги + сегмент таймфреймов (M15/M30/H1/D1) + «Скрыть». На узком
     экране 4-кнопочный .seg держал интринзик-ширину и сжимал имя в ноль → таймфреймы «перекрывали»
     название. Лечим как у instrument-preview-head: имя-колонка min-width:0 (эллипсит), а .seg —
     отдельной строкой во всю ширину под именем (flex-wrap), кнопки делят ширину поровну. Имя всегда
     читаемо. Десктоп (≥641px) не трогаем. */
  .gs-baskets .group-candidate-chart-head { flex-wrap: wrap; row-gap: 8px; }
  .gs-baskets .group-candidate-chart-head .instrument-preview-title { min-width: 0; flex: 1 1 auto; }
  .gs-baskets .group-candidate-chart-head .seg {
    order: 3; flex: 1 1 100%; min-width: 0;
    overflow-x: auto; -webkit-overflow-scrolling: touch; scrollbar-width: none;
  }
  .gs-baskets .group-candidate-chart-head .seg::-webkit-scrollbar { display: none; }
  .gs-baskets .group-candidate-chart-head .seg button { flex: 1 1 0; min-width: 64px; white-space: nowrap; }

  /* ===== Торговля: контент не должен залезать под нижний таб-бар =====
     Причина: корень экрана имеет min-height:100% + цепочку flex:1/min-height:0,
     из-за чего контейнер таблицы (flex:0 0 auto) ВЫЛЕЗАЕТ за бокс корня — последние
     строки рендерятся ниже бокса и НЕ удлиняют scrollHeight, поэтому padding-bottom
     внешнего .scroll до них не доезжает и они прячутся под .mobile-tabbar.
     На мобайле скроллит весь .scroll, поэтому распускаем фикс-высотную раскладку —
     экран течёт по высоте контента, и padding-bottom .scroll (74px) даёт реальный зазор.
     Десктоп (≥641px) не затрагивается. */
  .bonds-screen-root { min-height: auto !important; }
  .bonds-screen-main-row { flex: 0 0 auto !important; min-height: auto !important; align-items: stretch; }
  .bonds-screen-main-col { min-height: auto !important; }
  /* +зазор под таб-баром и для раскрытой строки (график/стакан добавляют высоту в конце списка) */
  .bonds-screen-table-wrap { min-height: auto !important; padding-bottom: 24px; }
}

/* ===== ТОРГОВЛЯ · мобильный свёрнутый вид (≤640): список бумаг → карточки .bcard,
   ряд групп → .mchip-пилюли. Эталон handoff/trading/mobile.dc.html стр.73-90 (карточка)
   и 74-76 (mchip-ряд). ДЕСКТОП (≥641) НЕ ТРОГАЕМ: таблица bonds-trade-table остаётся как есть. */
/* по умолчанию (десктоп) мобильные блоки спрятаны, видна таблица и десктоп-ряд вкладок */
.bonds-cards-mob, .bonds-grouptabs-mob, .bonds-screen-mobhead { display: none; }

/* Презентационные классы карточки из эталона (.numchip/.gchip/.cgrid/.kpi/.matrow/.mat/.mat-soon
   в проекте отсутствовали — заводим 1:1 как в макете; .mchip/.bcard уже есть выше). Скоупим
   .cgrid/.kpi через .bond-bcard, чтобы не задеть другие экраны. */
.numchip {
  flex: 0 0 auto; width: 19px; height: 19px; border-radius: 6px;
  background: var(--bg-neutral); color: var(--text-tertiary);
  display: grid; place-items: center;
  font-family: var(--font-mono); font-size: 10px; font-weight: 700;
}
.gchip {
  display: inline-flex; align-items: center; gap: 5px; padding: 3px 9px; border-radius: 999px;
  background: var(--bg-neutral); border: 1px solid var(--border-default);
  color: var(--text-secondary); font-size: 11px; font-weight: 600; white-space: nowrap; cursor: pointer;
}

@media (max-width: 640px) {
  /* десктоп-ряд вкладок групп и десктоп-таблицу — прячем; показываем мобильные блоки */
  .bonds-grouptabs-row, .bonds-trade-table-wrap { display: none !important; }
  .bonds-cards-mob { display: flex; flex-direction: column; gap: 10px; padding: 12px 14px 4px; min-width: 0; }
  .bonds-grouptabs-mob { display: block; padding: 10px 14px 0; min-width: 0; }

  /* --- ШАПКА ЭКРАНА «Торговля» (эталон mobile.dc.html стр.73): заголовок + ⚙Группа + чип робота.
     Баланс/Своб НЕ дублируем — он в общем топбаре приложения выше (реал-мани). --- */
  .bonds-screen-mobhead {
    display: flex; justify-content: space-between; align-items: center;
    gap: 8px; padding: 8px 14px 10px; min-width: 0;
  }
  .bonds-screen-mobhead-title { font-size: 20px; font-weight: 700; color: var(--text-primary); }
  .bonds-screen-mobhead .btn { flex: 0 0 auto; white-space: nowrap; }
  .bonds-screen-mobhead .tag { flex: 0 0 auto; white-space: nowrap; }

  /* --- ряд групп = горизонтальные пилюли (эталон стр.74-76) --- */
  .bonds-mchips {
    display: flex; gap: 8px; overflow-x: auto; padding-bottom: 10px;
    -webkit-overflow-scrolling: touch; scrollbar-width: none;
  }
  .bonds-mchips::-webkit-scrollbar { display: none; }
  .bonds-mchips .mchip { flex: 0 0 auto; }
  /* ⚙ компактно внутри активной пилюли — наследует контраст on-brand */
  .mchip-gear {
    display: inline-flex; align-items: center; justify-content: center;
    font-size: 13px; line-height: 1; margin-left: 1px; cursor: pointer;
    color: currentColor; opacity: .85; font-variant-emoji: text;
  }
  .mchip-gear:hover { opacity: 1; }
  /* extra-контролы (keep.md): +Новая / Глоб.правила / Портф.тестер — компактным рядом после пилюль */
  .bonds-mchips-actions {
    display: flex; gap: 6px; overflow-x: auto; padding-bottom: 10px;
    -webkit-overflow-scrolling: touch; scrollbar-width: none;
  }
  .bonds-mchips-actions::-webkit-scrollbar { display: none; }
  .bonds-mchips-actions .btn { flex: 0 0 auto; white-space: nowrap; }

  /* --- карточка бумаги (эталон стр.79-90) --- */
  .bond-bcard.active { border-color: var(--border-brand); }
  /* Раскрытая бумага: верхний (шапочный) блок подсвечивается зелёным тинтом — как на десктопе
     (table.t tbody tr.active > td { background: var(--bg-brand-tint) }). На мобайле разворот ставит
     .is-open, и зелёный ложится именно на .bond-bcard-tap (имя/KPI/matrow), а не на весь разворот. */
  .bond-bcard.is-open { border-color: var(--border-brand); }
  .bond-bcard.is-open .bond-bcard-tap { background: var(--bg-brand-tint); }
  .bond-bcard-tap { cursor: pointer; padding: 11px 12px; }
  .bond-bcard-head { display: flex; justify-content: space-between; align-items: flex-start; gap: 8px; }
  .bond-bcard-id { display: flex; align-items: flex-start; gap: 6px; min-width: 0; flex: 1; }
  /* имя бумаги — главный заголовок, кламп в 2 строки (длиннее → «…»); защита от наложения на статус */
  .bond-bcard-ticker {
    font-size: 14px; font-weight: 700; min-width: 0; flex: 1;
    display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical;
    overflow: hidden; line-height: 1.25; word-break: break-word; overflow-wrap: anywhere;
  }
  /* ★ + БС держим вместе справа от имени, прижатыми к верхней строке (не съезжают при 2-строчном имени) */
  .bond-bcard-id-badges { flex: 0 0 auto; display: inline-flex; align-items: center; gap: 6px; margin-top: 1px; }
  /* статус-чип защищён: режется «…», не давит на имя (title = полный текст) */
  .bond-bcard-status {
    flex: 0 1 auto; min-width: 0; max-width: 46%;
    overflow: hidden; text-overflow: ellipsis; white-space: nowrap; display: block;
  }
  .bond-bcard-sub {
    display: flex; justify-content: space-between; align-items: center; gap: 8px; margin-top: 6px;
  }
  .bond-bcard-issuer {
    font-size: 10px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; min-width: 0;
  }
  /* 8 KPI в 4 колонки */
  .bond-bcard .cgrid { display: grid; grid-template-columns: repeat(4, 1fr); gap: 7px 8px; margin-top: 10px; }
  .bond-bcard .kpi { display: flex; flex-direction: column; gap: 1px; min-width: 0; }
  .bond-bcard .kpi .k {
    font-size: 8.5px; font-family: var(--font-mono); text-transform: uppercase;
    letter-spacing: .03em; color: var(--text-tertiary);
  }
  .bond-bcard .kpi .v { font-size: 12px; font-weight: 600; font-family: var(--font-mono); white-space: nowrap; }
  /* matrow: плашка близости + auto/⟳/свитч */
  .bond-bcard .matrow {
    display: flex; align-items: center; justify-content: space-between; gap: 8px;
    margin-top: 9px; padding-top: 9px; border-top: 1px solid var(--border-default);
  }
  .bond-bcard .mat { font-family: var(--font-mono); font-size: 11px; color: var(--text-tertiary); }
  .bond-bcard .mat-soon {
    display: inline-flex; align-items: center; gap: 6px; padding: 3px 9px; border-radius: 999px;
    font-family: var(--font-mono); font-size: 10.5px; font-weight: 700;
    background: var(--bg-warning-tint); color: var(--text-warning); border: 1px solid var(--border-warning);
  }
  .bond-bcard .mat-soon.crit {
    background: var(--bg-error-tint); color: var(--text-error); border-color: var(--border-error);
  }
  /* разворот (график/стакан/ордера) — тот же контент, что в таблице; убираем 100vw-фикс таблицы */
  .bond-bcard-expand {
    border-top: 1px solid var(--border-brand); background: var(--bg-app);
  }
  .bond-bcard-expand .chart-inline-reveal { position: static; width: auto; max-width: 100%; }

  /* === ПУНКТ 1: «Заявки и ордера» карточками (эталон mobile.dc.html стр.102-109) === */
  /* На мобайле строка-аккордеон не должна иметь левый отступ 36px и быть row — делаем колонку. */
  .bond-orders-row { padding: 10px 12px 12px !important; flex-direction: column; }
  .bond-orders-row > .col { flex: 1 1 auto !important; }
  /* десктоп-грид прячем, карточки показываем */
  .bond-orders-row .orders-grid { display: none; }
  .bond-orders-cards { display: flex; flex-direction: column; gap: 6px; margin-top: 6px; }
  .bond-order-card {
    background: var(--bg-container); border: 1px solid var(--border-default);
    border-radius: 10px; padding: 8px 10px; min-width: 0;
  }
  .bond-order-card.slot { border-style: dashed; opacity: .55; }
  .bond-order-card .boc-top,
  .bond-order-card .boc-bottom,
  .bond-order-card .boc-meta {
    display: flex; justify-content: space-between; align-items: center; gap: 8px; min-width: 0;
  }
  .bond-order-card .boc-bottom { margin-top: 6px; }
  /* вторичная строка (статус ордера + время): прижата вправо, мельче; пустую не «распирает» */
  .bond-order-card .boc-meta { margin-top: 6px; justify-content: flex-end; gap: 8px; }
  .bond-order-card .boc-meta:empty { display: none; }
  .bond-order-card .boc-top > .tag,
  .bond-order-card .boc-meta > .tag { flex: 0 0 auto; }
  /* анимации появления/изменения/ухода — те же кадры, что у грид-строк (на карточке целиком) */
  .bond-order-card.tp-saved-flash { animation: tpSavedRowFlash 1.3s ease-out; }
  .bond-order-card.order-row-enter { animation: orderRowEnter .65s ease-out; }
  .bond-order-card.order-row-update { animation: orderRowUpdate .9s ease-out; }
  .bond-order-card.order-row-shrink { animation: orderRowShrink .95s ease-out; }
  .bond-order-card.order-row-leave { animation: orderRowLeave .85s ease forwards; pointer-events: none; }
  /* инлайн-инпут TP внутри карточки — нормальный тач-таргет (десктоп .tp-inline-input = 24px) */
  .bond-order-card .tp-inline-input { width: 64px; min-height: 34px; height: auto; padding: 5px 8px; text-align: right; }

  /* === ПУНКТ 3: «Настройки бумаги» полосой во всю ширину внизу разворота === */
  .bond-orders-settings-btn {
    display: flex; width: 100%; justify-content: center; min-height: 42px; margin-top: 12px;
  }
  /* кнопка «Настройки бумаги» в ШАПКЕ детали на мобайле скрыта (она теперь внизу) */
  .trd-settings-btn, .bond-yield-divider { display: none !important; }
}

/* ===== ДИЗАЙН-ПОЛИРОВКА (handoff дизайн-Клода): скроллбар, акцент строки, reduced-motion ===== */
/* Тонкий themed-скроллбар — класс .tbl-scroll вешается на обёртку с overflow (широкие таблицы/панели) */
.tbl-scroll { scrollbar-width: thin; scrollbar-color: var(--border-strong) transparent; }
.tbl-scroll::-webkit-scrollbar { height: 9px; }
.tbl-scroll::-webkit-scrollbar-track { background: transparent; }
.tbl-scroll::-webkit-scrollbar-thumb { background: var(--border-strong); border-radius: 999px; border: 2px solid var(--bg-app); }
.tbl-scroll::-webkit-scrollbar-thumb:hover { background: var(--text-tertiary); }

/* Универсальный тонкий тёмный скроллбар во всём приложении (фолбэк к class-scoped .tbl-scroll). */
* { scrollbar-width: thin; scrollbar-color: var(--border-strong) transparent; }
*::-webkit-scrollbar { width: 8px; height: 8px; }
*::-webkit-scrollbar-track { background: transparent; }
*::-webkit-scrollbar-thumb { background: var(--border-strong); border-radius: 6px; }
*::-webkit-scrollbar-thumb:hover { background: var(--text-tertiary); }

/* Ячейка «Погаш./Оферта» (дизайн-Клод): срок (.d) + строка даты (.s) с inline-чипом близости.
   ≤7 дн — crit (error), 8–20 дн — soon (warning) с пульсом; >20/нет дней — тихо.
   Пульс гасится глобальным prefers-reduced-motion ниже. */
.mat2 { display:flex; flex-direction:column; align-items:flex-end; line-height:1.18; }
.mat2 .d { font-weight:700; font-size:12px; }
.mat2 .s { font-size:9px; color:var(--text-tertiary); font-weight:500; }
.mat2.soon .d { color:var(--text-warning); }
.mat2.soon .s { color:var(--text-warning); font-weight:600; }
.mat2.crit .d { color:var(--text-error); }
.mat2.crit .s { color:var(--text-error); font-weight:600; }
.pchip { display:inline-flex; align-items:center; gap:4px; padding:1px 6px; border-radius:999px; font-family:var(--font-mono); font-size:8.5px; font-weight:700; }
.pchip.soon { background:var(--bg-warning-tint); color:var(--text-warning); }
.pchip.crit { background:var(--bg-error-tint); color:var(--text-error); }
.pulse { width:5px; height:5px; border-radius:50%; background:currentColor; animation:pl 1.4s ease-in-out infinite; }
@keyframes pl { 0%,100%{opacity:1;transform:scale(1);} 50%{opacity:.4;transform:scale(.7);} }

/* Цветной статус-акцент строки в таблице «Торговля» (--row-accent проставляет JSX по статусу).
   ВАЖНО: таблица — border-collapse, поэтому border-left на первой ячейке мёржится/клипается
   краями таблицы и липкой шапкой и красится обрывками. Используем inset box-shadow — он НЕ
   участвует в схлопывании границ и рисует ровную полосу на всю высоту поверх фона ячейки
   (переживает смену фона на hover/active). */
.bonds-trade-table tbody tr > td:first-child { box-shadow: inset 3px 0 0 0 var(--row-accent, transparent); }

/* «Позиции»: статус-акцент строки по источнику (робот/резерв/коррекция) — JSX ставит --row-accent.
   Скоупим строго на .pos-screen, чтобы не задеть другие .t-таблицы. Та же inset-техника. */
.pos-screen table.t tbody tr > td:first-child { box-shadow: inset 3px 0 0 0 var(--row-accent, transparent); }
/* «Позиции» (десктоп): длинное имя бумаги под тикером — одна строка с «…» (по эталону, max-width 150px).
   На мобайле td-head даёт white-space:normal, поэтому ограничиваем только вне мобильного брейка. */
@media (min-width: 641px) {
  .pos-screen table.t .pos-name {
    display: block; max-width: 160px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
  }
  /* Дружелюбное имя бумаги (1-я строка ячейки Инструмент) — одна строка с «…», как старый тикер. */
  .pos-screen table.t .pos-title {
    display: block; max-width: 180px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
    font-size: 13px;
  }
  /* Сырой ISIN после имени — приглушённый и мелкий (вторичный). */
  .pos-screen table.t .pos-rawcode { color: var(--text-tertiary); font-size: 10px; }
}

/* ===== «ПОЗИЦИИ» МОБАЙЛ (≤640): компактная KPI-панель + плотные карточки + неброские действия.
   Скоуп строго на .pos-screen, чтобы общий .t-cards / .kpi-grid и другие экраны не задеть.
   Эталон — handoff/screens/mobile.dc.html (карточка: имя+тикер слева, статус справа; 3-кол
   сетка данных; «К бумаге» = ghost). Состав полей/действий 1:1 — меняем только плотность/вид. */
@media (max-width: 640px) {
  /* --- ПУНКТ 1: KPI-панель компактнее (2 в ряд уже из общего правила; жмём вертикаль) --- */
  .pos-screen .kpi-grid .kpi-cell { padding: 8px 10px; gap: 2px; }
  .pos-screen .kpi-grid .kpi-label { font-size: 9px; letter-spacing: .02em; }
  .pos-screen .kpi-grid .kpi-value { font-size: 17px; line-height: 1.05; }
  .pos-screen .kpi-grid .kpi-sub { font-size: 9px; line-height: 1.15; }
  /* 5-я ячейка («Ждут коррекции») в 2-кол сетке висит одна — растягиваем на всю ширину,
     чтобы не было пустой клетки-дыры справа. */
  .pos-screen .kpi-grid .kpi-cell:nth-child(5) { grid-column: 1 / -1; }

  /* --- ПУНКТ 2: карточка позиции плотнее --- */
  .pos-screen .t-cards tbody tr {
    grid-template-columns: 1fr 1fr 1fr;   /* данные — в 3 колонки (эталон) */
    gap: 7px 10px; padding: 10px 11px; margin: 0 0 8px;
  }
  /* Шапка карточки: имя (с тикером) слева, статус-тег справа — в ОДНОЙ строке (grid-row:1).
     td-badge идёт в DOM раньше td-head, поэтому обоих явно сажаем на первую строку сетки. */
  .pos-screen .t-cards tbody td.td-head {
    grid-row: 1; grid-column: 1 / span 2; align-self: center; margin: 0; font-size: 13px;
    min-width: 0; overflow: hidden;
  }
  .pos-screen .t-cards tbody td.td-badge {
    grid-row: 1; grid-column: 3; justify-self: end; align-self: center; display: flex; margin: 0;
  }
  .pos-screen .pos-head-cell { min-width: 0; }
  .pos-screen .pos-title {
    display: block; font-weight: 600; font-size: 14px; line-height: 1.15;
    white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
  }
  .pos-screen .pos-name {
    display: block; font-size: 10px; line-height: 1.2;
    white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
  }
  .pos-screen .pos-rawcode { color: var(--text-tertiary); }
  /* Числовые поля: метка сверху, значение под ней (как .kpi в эталоне) — плотный столбик. */
  .pos-screen .t-cards tbody td[data-label] {
    flex-direction: column; align-items: flex-start; justify-content: flex-start;
    gap: 1px; white-space: nowrap;
  }
  .pos-screen .t-cards tbody td[data-label]::before {
    font-size: 9px; letter-spacing: .01em; color: var(--text-tertiary);
  }
  .pos-screen .t-cards tbody td[data-label] > * { font-size: 12px; }
  .pos-screen .t-cards tbody td.num[data-label] .col { align-items: flex-start !important; }

  /* --- ПУНКТ 3: действие — неброское, НЕ во всю ширину, чтобы не нажать случайно --- */
  .pos-screen .t-cards tbody td.td-action {
    grid-column: 1 / -1; display: flex; justify-content: flex-end; margin: 2px 0 0;
  }
  /* Перебиваем общий .t-cards .td-action .btn{width:100%} — у позиций кнопки авто-ширины. */
  .pos-screen .t-cards tbody td.td-action .btn {
    width: auto; min-height: 32px; padding: 0 12px; font-size: 12px;
  }
  .pos-screen .t-cards tbody td.td-action .pos-act-confirm { width: auto; }
  .pos-screen .t-cards tbody td.td-action .pos-act-confirm .btn { flex: 0 0 auto; }
  /* Пара кнопок коррекционной строки («К бумаге» + «В коррекцию») — как .pos-act-confirm:
     компактные, прижаты вправо, не растягиваются на всю ширину карточки. */
  .pos-screen .t-cards tbody td.td-action .pos-act-pair { width: auto; flex-wrap: wrap; justify-content: flex-end; }
  .pos-screen .t-cards tbody td.td-action .pos-act-pair .btn { flex: 0 0 auto; }
}

/* «Коррекция»: статус-акцент строки партии (частично исполнена / ручная партия) — JSX ставит
   --row-accent по реальному статусу. Скоуп строго на .corr-screen. Та же inset-техника. */
.corr-screen table.t tbody tr > td:first-child { box-shadow: inset 3px 0 0 0 var(--row-accent, transparent); }

/* «Коррекция» мобайл: таблица .t-cards → карточки. Порядок внутри карточки приводим к эталону
   (mobile.dc.html: тикер сверху → источник → поля Ордер/Куплено/Ждет/Цена/TP/Стоимость → кнопка).
   Десктоп (≥641) не затрагиваем — t-cards-классы активны только в этом медиазапросе. */
/* Модалка передачи партии (.scrim+.modal) — fixed-потомок .corr-screen. Корень несёт transform
   от входной анимации .fadein (translateY 4px), а любой transform у предка делает его containing
   block для fixed → scrim/боттом-шит считается от корня, а не от вьюпорта (на мобайле шит уезжал
   за экран; на десктопе модалка центрировалась в контент-области со смещением на ширину сайдбара).
   Лечим как .bonds-screen-root: анимация без сдвига (только opacity). Применяем на всех ширинах. */
.corr-screen.fadein { animation: fadein-noshift .18s ease-out; }

/* «Мои счета» хостит .scrim+.modal (добавление/удаление счёта) как fixed-потомков своего корня.
   Тот же кейс, что .corr-screen/.bonds-screen-root: входная .fadein несёт transform (translateY 4px),
   а любой transform у предка делает его containing block для fixed → scrim считается от корня, а не от
   вьюпорта (высокая модалка добавления уезжала шапкой за верх экрана). Лечим анимацией без сдвига. */
.accounts-screen.fadein { animation: fadein-noshift .18s ease-out; }

/* ===== КОСЯК 1 (БАЗА, не только мобайл): шапка карточки счёта — теги статуса (Алор / live /
   подключён) + чип RobotState («Робот работает» / «Отключен») в одной nowrap-строке НАЛЕЗАЛИ друг
   на друга и на десктопе тоже: карточки живут в сетке minmax(380px,1fr), т.е. ~380–550px шириной,
   и шапка не помещалась. Делаем перенос базовым правилом: чип подключения уходит ОТДЕЛЬНОЙ строкой
   под номером (flex-basis:100%), корзина держится справа на 1-й строке, строка тегов переносится.
   Перенос срабатывает только когда не влезает → на широкой одиночной карточке безвреден. ===== */
.accounts-screen .acct-card-head {
  flex-wrap: wrap;
  align-items: flex-start;
  column-gap: 10px;
  row-gap: 8px;
}
/* id-колонка занимает всю ширину между иконкой и корзиной (чип уходит вниз отдельной строкой) */
.accounts-screen .acct-card-head .acct-card-id { flex: 1 1 auto; }
/* строка тегов (номер · брокер · live · подключён) — ПЕРЕНОС вместо клипа */
.accounts-screen .acct-card-head .acct-card-tags { flex-wrap: wrap; row-gap: 4px; }
/* чип статуса подключения — на свою строку во всю ширину карточки, под блоком номера;
   order ставит его ПОСЛЕ корзины в визуальном порядке (DOM-порядок не трогаем) */
.accounts-screen .acct-card-head .acct-card-status {
  order: 3;
  flex: 1 0 100%;
  margin-top: 2px;
}
.accounts-screen .acct-card-head .acct-card-status > .tag { white-space: nowrap; }
/* корзина — фикс. справа на 1-й строке, не растягивается */
.accounts-screen .acct-card-head .acct-card-del {
  order: 2;
  flex: 0 0 auto;
  margin-left: auto;
  align-self: flex-start;
}

/* Карточки счетов: на десктопе сетка minmax(380px,1fr); на мобайле 380px-минимум вылезал бы за
   узкий вьюпорт (карточка шире контент-бокса) — на ≤640px складываем в одну колонку во всю ширину. */
@media (max-width: 640px) {
  .accounts-screen { padding: 14px !important; gap: 12px !important; }
  .accounts-screen .accounts-grid { grid-template-columns: 1fr !important; }

  /* ===== КОСЯК 1: базовая раскладка шапки (перенос + чип подключения отдельной строкой) теперь
     вынесена в базовое правило выше и работает на всех ширинах. На мобайле добавляем только дельту:
     увеличенный тач-таргет корзины (≥40px). ===== */
  .accounts-screen .acct-card-head .acct-card-del { min-height: 40px; }

  /* Биллинг (Фаза 3a): строка статуса лицензии/сервера — теги/чип «до даты» переносятся опрятно,
     без клипа/вылета; подпись «Доступ» не сжимается в столбик. */
  .accounts-screen .acct-card-license { row-gap: 4px; }
  .accounts-screen .acct-card-license > .tag { white-space: nowrap; }
  .accounts-screen .acct-card-license-until { white-space: nowrap; }

  /* Строка статуса токена — теги/кнопка переносятся опрятно на узком экране; кнопка тянется по ширине. */
  .accounts-screen .acct-card-token { row-gap: 6px; }
  .accounts-screen .acct-card-token .acct-token-btn { flex: 1 1 auto; justify-content: center; }

  /* ===== КОСЯК 2: модалка «Регистрация у брокера…» (полноэкранный лист — дженерик .modal
     боттом-шит уже даёт width:100% / border-radius сверху). Доводим внутренности: ===== */
  /* видео-плейсхолдер во всю ширину листа (тело уже padding:16) — aspect-ratio 16/9 держит высоту */
  .add-account-modal .add-account-video { width: 100%; }
  .add-account-modal .add-account-video-play { width: 60px; height: 60px; }

  /* «Будет добавлено»: на десктопе label слева + теги справа (space-between); на узком экране
     это сжимало теги/маску токена → ставим столбиком, теги переносятся опрятно (wrap) */
  .add-account-modal .add-account-preview {
    flex-direction: column;
    align-items: stretch;
    gap: 8px;
  }
  .add-account-modal .add-account-preview > .row.wrap { flex-wrap: wrap; row-gap: 6px; }

  /* шаги: ссылка не вылезает за лист (длинный URL ломаем по символам, как в mobile.dc.html) */
  .add-account-modal .add-account-step-link { word-break: break-all; }

  /* футер: примечание про администратора + кнопки в одну строку «наезжали» (note сжимался
     в 4 строки, кнопки лепились вплотную). Раскладываем ВЕРТИКАЛЬНО: примечание сверху,
     кнопки ниже в ряд во всю ширину (тач-таргет ≥44px из дженерика .btn). */
  .add-account-modal .add-account-foot {
    flex-direction: column;
    align-items: stretch;
    gap: 10px;
  }
  .add-account-modal .add-account-foot-note { order: -1; }
  .add-account-modal .add-account-foot-actions { width: 100%; }
  .add-account-modal .add-account-foot-actions .btn { flex: 1 1 0; justify-content: center; }
}

@media (max-width: 640px) {
  /* акцент статуса — левой кромкой карточки (inset на td:first-child тут не к месту: первым
     в DOM идёт «Ордер», а не видимо-первый тикер) */
  .corr-screen .t-cards tbody tr { border-left: 3px solid var(--row-accent, var(--border-default)); }
  .corr-screen .t-cards tbody tr > td:first-child { box-shadow: none; }
  /* визуальный порядок ячеек в карточке (DOM-порядок десктопа сохранён) */
  .corr-screen .t-cards tbody td.td-head { order: -3; }
  .corr-screen .t-cards tbody td.td-badge { order: -2; }
  .corr-screen .t-cards tbody td[data-label="Ордер"] { order: -1; grid-column: 1 / -1; }
  .corr-screen .t-cards tbody td.td-action { order: 10; }
  /* две кнопки действия (Робот / В резерв) делят ширину карточки поровну */
  .corr-screen .t-cards tbody td.td-action .row { width: 100%; }
  .corr-screen .t-cards tbody td.td-action .btn { flex: 1 1 0; justify-content: center; min-height: 40px; }
}

/* «Тестеры»: строка «Итого»/«Итог портфеля» — жирная и с верхней линией, чтобы отделить от строк. */
table.t.tester-results-table tbody tr.tester-total-row > td { font-weight: 700; border-top: 2px solid var(--border-strong); }

/* «История»: строка-выплата купона — деликатный info-акцент (кромка + лёгкий тон), JSX ставит --row-accent.
   Та же inset box-shadow техника (не клипается под border-collapse). */
.hist-screen table.t tbody tr > td:first-child { box-shadow: inset 3px 0 0 0 var(--row-accent, transparent); }
.hist-screen table.t tbody tr.hist-coupon-row > td { background: oklch(from var(--color-info-500) l c h / 0.06); }
/* «История»: «Чистая» — итоговая сумма, жирная (по эталону: font-weight:700, цвет по знаку). */
.hist-screen table.t .hist-net { font-weight: 700; }
/* «История»: «Итог с купоном» (PNL-2) — полный результат с купоном, тоже жирный (headline-итог). */
.hist-screen table.t .hist-net-coupon { font-weight: 700; }
/* «История» (десктоп): длинное имя бумаги под тикером — одна строка с «…». На мобайле td-head
   распускает имя (white-space:normal), поэтому ограничиваем только вне мобильного брейка. */
@media (min-width: 641px) {
  .hist-screen table.t .hist-name {
    display: block; max-width: 220px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
  }
  /* Свёрнутая шапка карточки — артефакт мобильной раскладки; на десктопе таблица показывает все
     ячейки построчно, а эта сводка не нужна. Полностью убираем из потока таблицы. */
  .hist-screen table.t .hist-card-summary { display: none; }
}
/* ===== «ИСТОРИЯ» МОБАЙЛ (≤640): компактная KPI-панель, опрятная шапка и КАРТОЧКИ С РАСКРЫТИЕМ.
   Скоуп строго на .hist-screen, чтобы общий .t-cards / .kpi-grid и другие экраны не задеть.
   Эталон — handoff/screens/mobile.dc.html (История): свёрнутая карточка = название слева + сумма
   справа; по тапу — поля Покупка/Продажа/Лот/Комиссия/Прибыль/Купон/Чистая/Источник. Десктоп ≥641
   не трогаем. Состав полей 1:1 (всё доступно в раскрытом виде) + 2 типа строк (сделка/купон). */
@media (max-width: 640px) {
  /* --- ПУНКТ 1а: шапка экрана столбиком (как в «Позициях»): заголовок сверху, управление —
     отдельным рядом под ним во всю ширину (иначе кнопка+поиск+4-сегмент разносило по краям). --- */
  .hist-screen .hist-head { flex-direction: column; align-items: stretch !important; gap: 10px; }
  .hist-screen .hist-head-ctl { flex-wrap: wrap; gap: 8px; }
  /* поиск — во всю ширину (убиваем inline width:220) */
  .hist-screen .hist-head-ctl .hist-search { flex: 1 1 100%; width: 100% !important; min-width: 0; }
  /* сегмент периода — во всю ширину, 4 кнопки делят поровну (под палец) */
  .hist-screen .hist-head-ctl .seg { width: 100%; }
  .hist-screen .hist-head-ctl .seg button { flex: 1 1 0; }
  /* кнопка «Обновить» не растягивается на всю строку — авто-ширина слева */
  .hist-screen .hist-head-ctl > .btn.sm { min-height: 36px; }

  /* --- ПУНКТ 1б: KPI-панель компактнее (2 в ряд из общего правила; жмём вертикаль) --- */
  .hist-screen .kpi-grid .kpi-cell { padding: 8px 10px; gap: 2px; }
  .hist-screen .kpi-grid .kpi-label { font-size: 9px; letter-spacing: .02em; }
  .hist-screen .kpi-grid .kpi-value { font-size: 17px; line-height: 1.05; }
  .hist-screen .kpi-grid .kpi-sub { font-size: 9px; line-height: 1.15; }
  /* 5-я ячейка («Купонный доход») в 2-кол сетке висит одна — растягиваем во всю ширину,
     чтобы не было пустой клетки-дыры справа (тот же приём, что в «Позициях»). */
  .hist-screen .kpi-grid .kpi-cell:nth-child(5) { grid-column: 1 / -1; }

  /* --- ПУНКТ 2: КАРТОЧКИ С РАСКРЫТИЕМ ---
     Свёрнуто показываем ТОЛЬКО сводку (.hist-card-summary): прячем все td строки, затем
     возвращаем сводку. Раскрыто (.is-open) — показываем поля деталей в их обычной мобильной
     раскладке; дублирующую шапку с именем (td-head) держим скрытой, имя есть в сводке. */
  .hist-screen .t-cards tbody tr.hist-row { padding: 0; gap: 0; overflow: hidden; }
  .hist-screen .t-cards tbody tr.hist-row > td { display: none; }
  .hist-screen .t-cards tbody tr.hist-row > td.hist-card-summary { display: flex; }

  /* свёрнутая шапка = одна компактная строка во всю ширину, тап-таргет ≥44px */
  .hist-screen .t-cards tbody td.hist-card-summary {
    grid-column: 1 / -1;
    display: flex; align-items: center; justify-content: space-between; gap: 10px;
    min-height: 48px; padding: 11px 13px; margin: 0; cursor: pointer;
    white-space: nowrap; overflow: hidden;
  }
  .hist-screen .t-cards tbody td.hist-card-summary::before { content: none; }
  .hist-card-summary .hist-sum-main {
    display: flex; align-items: baseline; gap: 6px; min-width: 0; overflow: hidden;
  }
  .hist-card-summary .hist-sum-title {
    font-size: 14px; font-weight: 600; color: var(--text-primary);
    white-space: nowrap; overflow: hidden; text-overflow: ellipsis; min-width: 0;
  }
  .hist-card-summary .hist-sum-raw {
    font-size: 10px; color: var(--text-tertiary); flex-shrink: 0;
  }
  .hist-card-summary .hist-sum-cpntag { flex-shrink: 0; padding: 0 6px; }
  .hist-card-summary .hist-sum-right { display: flex; align-items: center; gap: 8px; flex-shrink: 0; }
  .hist-card-summary .hist-sum-net { font-size: 14px; font-weight: 700; }
  .hist-card-summary .hist-sum-net.pos { color: var(--text-success); }
  .hist-card-summary .hist-sum-net.neg { color: var(--text-error); }
  .hist-card-summary .hist-sum-net.muted { color: var(--text-tertiary); }
  .hist-card-summary .hist-sum-chev {
    display: inline-flex; color: var(--text-tertiary); transition: transform .15s ease;
  }
  .hist-card-summary .hist-sum-chev.is-open { transform: rotate(180deg); }

  /* раскрыто: вернуть поля деталей; внутренний отступ карточки восстановить (сводка свой имеет) */
  .hist-screen .t-cards tbody tr.hist-row.is-open { gap: 6px 16px; }
  .hist-screen .t-cards tbody tr.hist-row.is-open > td[data-label] { display: flex; }
  .hist-screen .t-cards tbody tr.hist-row.is-open > td.td-coupon-note { display: flex; }
  .hist-screen .t-cards tbody tr.hist-row.is-open > td.td-source { display: flex; }
  /* имя уже показано в сводке — дублирующую td-head в раскрытии не показываем */
  .hist-screen .t-cards tbody tr.hist-row.is-open > td.td-head { display: none; }
  /* отбивка блока деталей от шапки-сводки сверху (лёгкая линия), боковые/нижний паддинг полей */
  .hist-screen .t-cards tbody tr.hist-row.is-open > td[data-label],
  .hist-screen .t-cards tbody tr.hist-row.is-open > td.td-coupon-note,
  .hist-screen .t-cards tbody tr.hist-row.is-open > td.td-source { padding-left: 13px; padding-right: 13px; }
  .hist-screen .t-cards tbody tr.hist-row.is-open > td[data-label]:first-of-type { margin-top: 4px; }
  .hist-screen .t-cards tbody tr.hist-row.is-open > td.td-source { padding-bottom: 12px; }

  /* строка-выплата купона: левый info-акцент кромкой + без инлайн-фона ячеек (как было) */
  .hist-screen .t-cards tbody tr.hist-coupon-row { border-left: 3px solid var(--color-info-500); }
  .hist-screen .t-cards tbody tr.hist-coupon-row > td { background: transparent; }
  .hist-screen .t-cards tbody td.td-coupon-note {
    grid-column: 1 / -1; align-items: center; gap: 6px;
    justify-content: flex-start; white-space: normal; text-align: left;
    color: var(--text-info); margin: 2px 0 0;
  }
  .hist-screen .t-cards tbody td.td-coupon-note::before { content: none; }
  /* источник-тег — во всю ширину снизу карточки */
  .hist-screen .t-cards tbody td.td-source { grid-column: 1 / -1; justify-content: flex-start; margin: 4px 0 0; }
}

/* Доступность: гасим анимации для чувствительных к движению */
@media (prefers-reduced-motion: reduce) {
  *, *::before, *::after {
    animation-duration: .001ms !important;
    animation-iteration-count: 1 !important;
    transition-duration: .001ms !important;
    scroll-behavior: auto !important;
  }
  .btn:active { transform: none; }
}

/* =============================================================================
   «ТОРГОВЛЯ» — глубокий пас под макет trading-redesign (презентационный).
   Только существующие токены; разметка/данные/логика не тронуты.
   ========================================================================== */

/* --- Липкая шапка таблицы: фон на тон светлее контейнера (макет: --bg-container-2),
   чтобы заголовки отделялись от строк при прокрутке. Поверх z над акцент-кромкой строк. --- */
table.t.bonds-trade-table thead th {
  background: var(--bg-container-2);
  z-index: 3;
  border-bottom-color: var(--border-strong);
}

/* --- Плотность и читаемость строк (макет brow): компактнее по вертикали,
   чёткий ховер и выбранная строка, цена/ключевые цифры контрастнее. --- */
table.t.bonds-trade-table tbody td { padding-top: 8px; padding-bottom: 8px; }
table.t.bonds-trade-table tbody tr:not(.chart-inline-row):hover > td { background: var(--bg-neutral); }
table.t.bonds-trade-table tbody tr.active > td { background: var(--bg-brand-tint); }
table.t.bonds-trade-table tbody tr.active:hover > td { background: var(--bg-brand-tint); }
/* Раскрытая строка (график + ордера) — это контент, а не «наводимая» строка данных.
   Гасим ховер-подсветку и на уровне tr, и на уровне td: иначе при движении мыши над
   графиком/ордерами фон строки переключается на --bg-neutral с transition .12s и поверх
   живого графика читается как уродливое мигание. Держим фон стабильным (прозрачным). */
.bonds-trade-table tbody tr.chart-inline-row:hover,
.bonds-trade-table tbody tr.chart-inline-row:hover > td { background: transparent; }
/* Цена — якорь строки: чуть контрастнее остальных чисел. */
table.t.bonds-trade-table tbody td.num:nth-of-type(4) { color: var(--text-primary); }

/* Фильтр-статусы теперь рисуются ПРОСТЫМ .seg (макет ~161: «Все статусы / Торгуются / Жду цену /
   Пауза» — обычный сегмент-контрол, не «таблетки»). Прежняя обёртка .fchips-wrap с пилюлями
   удалена; .seg использует базовый стиль из секции «Segmented control». */

/* --- Тумблер «Торг» (макет trd-toggle): крупнее обычного .switch (40×22 vs 32×18) под палец,
   on — зелёный с белым кружком. Скоупим строго на ячейку .trd-cell торговой таблицы. --- */
.bonds-trade-table td.trd-cell .switch {
  width: 40px;
  height: 22px;
  border: 1px solid var(--border-strong);
}
.bonds-trade-table td.trd-cell .switch::after {
  width: 16px;
  height: 16px;
  left: 2px;
  /* 16px кружок в 22px дорожке → top 3px центрирует по вертикали (3+16+3=22). */
  top: 3px;
  background: var(--text-secondary);
}
.bonds-trade-table td.trd-cell .switch.on {
  border-color: var(--bg-brand);
}
.bonds-trade-table td.trd-cell .switch.on::after {
  /* симметричный зазор: 40-16-2-2=20px (кружок встаёт в 2px от правого края). */
  transform: translateX(20px);
  background: var(--text-on-brand);
}

/* --- Стакан: средняя цена по центру крупнее и жирнее (макет: 13px/700) + рамка ярче. --- */
.orderbook-mid { border-color: var(--border-strong); }
.orderbook-mid .mono:first-child { font-size: 13px; font-weight: 700; color: var(--text-primary); }

/* --- Лента/секции аккордеона: заголовок «Заявки и открытые ордера» — чуть крупнее счётчик,
   плюс верхняя кромка-разделитель у блока ордеров (секционное деление как в макете). --- */
.orders-grid-head > span { padding-top: 7px; padding-bottom: 7px; }

/* --- Чип-счётчик в eyebrow-заголовках раскрытой строки (макет: круглый бейдж рядом с заголовком). --- */
.section-count {
  display: inline-grid;
  place-items: center;
  min-width: 18px;
  height: 16px;
  padding: 0 5px;
  background: var(--bg-neutral);
  color: var(--text-secondary);
  border-radius: 999px;
  font-size: 10px;
  font-family: var(--font-mono);
  font-weight: 600;
}

/* --- Раскрытая строка бумаги: брендовая верхняя кромка (макет: border-top var(--border-brand)),
   чтобы развёрнутый блок графика/стакана/ордеров читался как «продолжение выбранной строки».
   Фон всего раскрытия — --bg-container (макет: td раскрытия на var(--bg-container)); полосы
   шапки/тулбара/стакана/ордеров уже на --bg-container, а зона под графиком (обёртка списка
   ордеров) раньше просвечивала --bg-app — теперь тоже серая, как в макете. --- */
.chart-inline-reveal { border-top: 1px solid var(--border-brand); background: var(--bg-container); }
/* Тулбар графика и шапка стакана внутри раскрытия — фон-контейнер, чтобы секции отделялись
   от тела графика (макет: каждая полоса детали на --bg-container с нижней границей). */
.chart-inline-stage .chart-layer-switches { align-items: center; }

/* =============================================================================
   «ТОРГОВЛЯ» — пас 2: отвязка панели настроек от inline-графика + правки вёрстки.
   Только презентация; токены; разметка/данные/логика не тронуты.
   ========================================================================== */

/* --- Кнопка «Настройки бумаги» в нажатом состоянии, пока открыт правый drawer (settingsOpen).
   Делает очевидной связь «верхняя кнопка → открытая панель»: выглядит вдавленной (brand-press
   + inset-тень), без перекрытия дефолтных primary-теней. Поведение кнопки не меняется. --- */
.btn.primary.trd-settings-btn.active {
  background: var(--bg-brand-press);
  border-color: var(--bg-brand-press);
  box-shadow: inset 0 1px 3px oklch(0 0 0 / 0.28);
}
.btn.primary.trd-settings-btn.active:hover {
  background: var(--bg-brand-press);
  border-color: var(--bg-brand-press);
  box-shadow: inset 0 1px 3px oklch(0 0 0 / 0.28);
}

/* --- Правый drawer настроек как явно отдельная плавающая панель над страницей.
   borderLeft (2px var(--border-strong)) и boxShadow (var(--shadow-modal)) заданы инлайном;
   здесь только гарантируем, что панель рисует собственный контекст наложения над страницей
   (стек выше акцент-теней строк), без скрима/затемнения графика слева. --- */
.bond-side-panel { isolation: isolate; }

/* Шапка панели на ВСЕХ ширинах: при 360px десктоп-drawer шапка — нешироприносящий flex-row
   (justify-content:space-between инлайн), кнопки «Удалить/Заменить»+«Скрыть» не сжимались
   (actions flexShrink:0 инлайн) и при длинном тикере/имени вылезали за правый край панели →
   «Удалить» был не виден на десктопе. Разрешаем шапке переносить кнопки под заголовок, а имя-
   колонке — ужиматься/эллипсить. flex-wrap срабатывает только при нехватке ширины, поэтому при
   коротком тикере/имени раскладка не меняется (без регресса). Раньше эти правила жили только в
   @media (max-width:640px) ниже — теперь они база, мобильные дубли стали безвредными оверрайдами. */
.bond-side-panel-head { flex-wrap: wrap; row-gap: 8px; }
.bond-side-panel-head-title { min-width: 0; flex: 1 1 auto; }
.bond-side-panel-head-title .eyebrow { overflow: hidden; text-overflow: ellipsis; }
.bond-side-panel-head-actions { flex-wrap: wrap; justify-content: flex-end; }
/* Дублирующий ряд действий [← Назад][Удалить/Заменить] нужен ТОЛЬКО на мобайле (там шапка панели
   уезжает за верх экрана). На десктопе кнопки живут в шапке-сайдбаре справа — этот ряд скрыт. */
.bond-side-panel-mobile-actions { display: none; }

/* мобайл «Настройки бумаги»: панель во ВЕСЬ экран (десктоп — 360px drawer с top:48, не трогаем) */
@media (max-width: 640px) {
  /* ПУНКТ 4: панель .bond-side-panel — position:fixed, но её предок .bonds-screen-root
     несёт transform от входной анимации .fadein (translateY 4px). Любой transform у предка
     делает его «containing block» для fixed-потомка → панель привязывается к прокрученному
     корню (top уезжает на offset корня, height:100% = высота корня, а не вьюпорта), и лист
     не покрывает топбар. На мобайле убираем у корня transform-составляющую анимации (оставляем
     только opacity-fade) — фикс-панель снова считается от вьюпорта. Десктоп (≥641) не трогаем. */
  .bonds-screen-root.fadein { animation: fadein-noshift .18s ease-out; }

  .bond-side-panel {
    width: auto !important;
    left: 0 !important;
    right: 0 !important;
    border-left: 0 !important;
    border-radius: 0 !important;
    /* ПУНКТ 4 (эталон mobile.dc.html стр.190: лист height:100%, margin-top:0) — снимаем
       инлайн top:48 (зазор под топбаром), растягиваем на всю высоту как полноэкранный лист.
       100dvh — динамическая высота вьюпорта (учёт адресной строки мобильного браузера). */
    top: 0 !important;
    bottom: 0 !important;
    height: 100dvh !important;
  }
  /* тач-таргеты полей внутри панели (как в макете: 40px) */
  .bond-side-panel .input,
  .bond-side-panel .select {
    min-height: 40px;
  }
  /* Поля настроек бумаги — парами по 2 в ряд (эталон mobile.dc.html стр.197-201: Период+Коэф,
     Шаг+Ордеров, Сетки+Отступ и т.д.). Общий блок выше сбрасывает .settings-grid в 1 колонку
     (профиль/тестер), но в полноэкранной панели «Настройки бумаги» узкие числовые поля влезают
     2-в-ряд на 360px и так компактнее/как в макете. Длинные подписи переносятся (.settings-field
     .field-label white-space:normal + min-height:24px держит ряды ровными). Десктоп не трогаем. */
  .bond-side-panel .settings-grid { grid-template-columns: 1fr 1fr !important; column-gap: 10px; }
  /* Шапка панели: имя-колонка ужимается и эллипсит (был жёсткий max-width:220 + nowrap-eyebrow —
     при длинном тикере/имени шапка распирала панель и кнопки «Удалить/Скрыть» вылезали). Имя-колонка
     min-width:0, кнопки справа не сжимаются и переносятся под заголовок, если не хватает ширины. */
  .bond-side-panel-head { flex-wrap: wrap; row-gap: 8px; }
  .bond-side-panel-head-title { min-width: 0; flex: 1 1 auto; }
  .bond-side-panel-head-title .eyebrow { white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
  .bond-side-panel-head-name { max-width: 100% !important; }
  .bond-side-panel-head-actions { flex-wrap: wrap; justify-content: flex-end; }
  .bond-side-panel-head-actions .btn.sm { min-height: 36px; }
  /* липкая нижняя панель «Сохранить/Сбросить» — только на мобайле.
     Таб-бар на время открытой панели скрыт (см. правило с :has ниже), поэтому savebar садится
     на истинный низ вьюпорта — добавляем safe-area-inset в нижний паддинг, чтобы кнопки не
     попали под жест-бар/«домашний» индикатор. */
  .bond-side-panel .bond-settings-savebar {
    position: sticky; bottom: 0; z-index: 2;
    background: var(--bg-container);
    border-top: 1px solid var(--border-default);
    padding: 10px 0 calc(10px + env(safe-area-inset-bottom, 0px)); margin-top: 4px;
  }
  /* Липкий ряд действий [← Назад][Удалить/Заменить] — прибит к низу панели тем же sticky-bottom
     механизмом, что и savebar (он у владельца работает). Панель — flex-колонка height:100dvh,
     .scroll выше тянется (flex:1), а этот ряд не сжимается и держится у низа экрана. Лежит НАД
     savebar «Сохранить/Сбросить» (savebar — внутри .scroll вкладки «Настройки»; этот ряд —
     прямой ребёнок панели, виден на всех вкладках). Компактно для 375px: иконка + короткий текст,
     кнопки не переносятся (flex-shrink:0 + marginLeft:auto разводит их по краям). */
  .bond-side-panel-mobile-actions {
    display: flex; flex-shrink: 0;
    position: sticky; bottom: 0; z-index: 3;
    align-items: center;
    padding: 8px 14px calc(8px + env(safe-area-inset-bottom, 0px));
    background: var(--bg-container);
    border-top: 1px solid var(--border-default);
  }
  .bond-side-panel-mobile-actions .btn.sm { min-height: 36px; }
}

/* =================================================================
   СИСТЕМНЫЙ ФИКС: прячем нижний таб-бар, пока открыт ЛЮБОЙ полноэкранный оверлей.
   Зачем: фикс. таб-бар (.mobile-tabbar, z:90, bottom:0) перекрывал нижние кнопки/savebar/футеры
   оверлеев — и тех, что лежат НИЖЕ него по стеку (полноэкранная панель «Настройки бумаги»
   .bond-side-panel, z:40), и тех, что формально выше (модалки .scrim z:100 — но в этой
   раскладке таб-бар всё равно перехватывал клики по кнопке у самого низа: «Готово» портф.
   тестера попадал в иконку таб-бара). Любой такой оверлей на мобайле — модал во весь экран,
   нижняя навигация в этот момент не нужна и только мешает.
   Решение одним правилом для ВСЕХ кейсов сразу (модалки ЧС/счёта/удаления/коррекции/справочника/
   глоб.правил/портф.тестера/полного тестера + панель «Настройки бумаги»): пока в DOM есть .scrim
   ИЛИ .bond-side-panel — таб-бар display:none. Закрылся оверлёй — таб-бар возвращается.
   Только ≤640px (на десктопе таб-бара нет вовсе). Десктоп ≥641 не затрагивается. ================ */
@media (max-width: 640px) {
  body:has(.scrim) .mobile-tabbar,
  body:has(.bond-side-panel) .mobile-tabbar { display: none !important; }
}

/* --- Кластер доходностей в шапке детали (макет ~223-229): справа от тикера, 4 ячейки
   «К погаш. / Купон / Цена / Оферта» — 9px-лейбл + mono-значение, выровнены вправо.
   К погаш. — обычным цветом (акцент), остальные muted (заданы классом в разметке). --- */
.bond-yield-cluster { flex-wrap: wrap; align-items: center; }
.bond-yield-cluster .byc-cell { gap: 0; text-align: right; align-items: flex-end; }
.bond-yield-cluster .byc-cell .label { font-size: 9px; }
.bond-yield-cluster .byc-val { font-weight: 600; }
/* ПУНКТ 2 (эталон mobile.dc.html стр.101): на узкой ширине (≤1100, открыт правый drawer /
   планшет / телефон) горизонтальный кластер не помещается рядом с кнопкой. Раньше его прятали
   совсем; теперь — раскладываем СЕТКОЙ 4×1 (как .cgrid в эталоне): «К погаш./Купон/Цена/Оферта»,
   на всю ширину под тикером. К погаш. жирн., Купон/Оферта tertiary, Цена — success. Десктоп
   ≥1101 не трогаем (остаётся инлайн справа в шапке). Разделитель на узком прячем. */
@media (max-width: 1100px) {
  .bond-yield-divider { display: none; }
  .bond-yield-cluster {
    display: grid !important; grid-template-columns: repeat(4, 1fr);
    gap: 7px 8px; flex: 1 1 100% !important; width: 100%; margin-top: 2px;
  }
  .bond-yield-cluster .byc-cell {
    flex-direction: column; align-items: flex-start; text-align: left; gap: 1px; min-width: 0;
  }
  .bond-yield-cluster .byc-cell .label {
    font-size: 8.5px; font-family: var(--font-mono); text-transform: uppercase;
    letter-spacing: .03em; color: var(--text-tertiary);
  }
  .bond-yield-cluster .byc-val { font-size: 12px; font-weight: 600; white-space: nowrap; }
  /* первая ячейка (К погаш.) — акцентно жирная; «Цена» — зелёная (эталон), как в .cgrid разворота */
  .bond-yield-cluster .byc-cell:first-child .byc-val { font-weight: 700; }
  .bond-yield-cluster .byc-cell:nth-child(3) .byc-val { color: var(--text-success); }
}

/* --- Липкая шапка торговой таблицы: акцент-полоса строки — inset box-shadow на td:first-child,
   шапка же построена на th, поэтому акцент в неё не «протекает». На всякий случай гасим любую
   тень у заголовочных ячеек и держим непрозрачный фон, чтобы при прокрутке не было щели. --- */
table.t.bonds-trade-table thead th:first-child { box-shadow: none; }

/* ============================================================
   Настройки ИНВЕСТ-группы — компоновка точ-в-точ по handoff/invest-group/*.dc.html.
   Стек карточек на наших токенах/классах (.block/.tag/.switch/.input/.select). Спекулятивную
   ветку (.group-detail-body grid) НЕ трогаем — у инвеста отдельный контейнер .gs-invest.
   ============================================================ */
/* Прогресс-бар распределения весов ([NEW] по spec.md). */
.wbar { height: 8px; border-radius: 999px; background: var(--bg-neutral); overflow: hidden; }
.wbar > div { transition: width .2s ease; }
/* Ряд-тумблер общих настроек (как в эталоне .toggle-row). */
.gs-invest .toggle-row {
  display: flex; justify-content: space-between; align-items: center;
  gap: 10px; padding: 11px 0; border-bottom: 1px solid var(--border-default);
}
.gs-invest .toggle-row:last-child { border-bottom: 0; }
/* Липкий savebar инвеста (управляет скалярными настройками). */
.gs-invest-savebar {
  position: sticky; bottom: 0; z-index: 20;
  display: flex; align-items: center; gap: 10px;
  padding: 12px 16px; margin-top: 0;
  background: var(--bg-container); border: 1px solid var(--border-default);
  border-radius: 8px; box-shadow: 0 -8px 20px rgba(0, 0, 0, .28);
}
/* Поля шапки (лимит/дюрация) — фикс. ширина на десктопе как в эталоне (200px). */
.gs-invest-field { width: 200px; max-width: 100%; }
/* Хлебные крошки видны на десктопе, кнопка «Назад» (дублёр) — только на мобайле. */
.gs-invest-back { display: none; }

@media (max-width: 640px) {
  .gs-invest { padding: 14px !important; gap: 12px !important; }
  /* На мобайле прячем десктопные крошки, показываем кнопку «Назад» (эталон mobile стр.41). */
  .gs-invest-crumb { display: none; }
  .gs-invest-back { display: inline-flex; }
  /* Десктопные таблицы состава/ЧС прячем, показываем карточки (.gs-mob-cards включается дженериком). */
  .gs-invest .gs-desk-only { display: none !important; }
  /* Карточки получают внутренний паддинг от .block-body padding=0 → задаём свой. */
  .gs-invest .gs-mob-cards { padding: 12px; }
  /* Шапка: имя/тумблеры/поля во всю ширину, без горизонт-вылета. */
  .gs-invest-head-row { flex-direction: column; align-items: stretch !important; gap: 10px !important; }
  .gs-invest-head-toggles { justify-content: flex-start; gap: 24px !important; }
  .gs-invest-head-fields { flex-direction: column; align-items: stretch !important; }
  .gs-invest-field { width: 100% !important; }
  .gs-invest .gs-invest-field .input,
  .gs-invest .gs-invest-field .select { width: 100%; }
  .gs-invest-limit-hint { order: 3; }
  .gs-invest .h1 { white-space: normal; font-size: 19px; }
  /* Savebar внизу карты, как в эталоне (липкий над таб-баром приложения). */
  .gs-invest-savebar { bottom: 0; padding: 11px 14px calc(11px + env(safe-area-inset-bottom, 0px)); }
  .gs-invest-savebar .btn { min-height: 42px; }
}

/* ============================================================
   Партнёрская программа (screen-partners.jsx) + модалка «Продлить сервер»
   (dialog-server-extend.jsx). Тёмная тема, переиспользуем .block/.btn/.tag.
   Золото (--text-money) — только на суммах. @media ≤640 — таблицы в карточки.
   ============================================================ */
/* Экран — раскладка как в эталоне: вертикальная колонка с gap, паддинг страницы. */
.partners-screen { max-width: 1180px; width: 100%; margin: 0 auto; box-sizing: border-box; padding: 24px; gap: 16px; }
.partners-screen .money { color: var(--text-money); font-family: var(--font-mono); font-feature-settings: "tnum" 1, "lnum" 1; font-weight: 700; }
.partners-sub { margin: 0; color: var(--text-tertiary); font-size: 13px; }

/* HERO «витрина» — реф-баланс крупно золотом, акцент-градиент в углу */
.partners-hero { background: radial-gradient(120% 130% at 82% 0%, rgba(31,138,91,0.20), transparent 56%), var(--color-neutral-950); border: 1px solid var(--border-strong); border-radius: 18px; padding: 22px; position: relative; overflow: hidden; }
.partners-hero-eyebrow { font-size: 10px; letter-spacing: .14em; color: var(--text-tertiary); }
.partners-hero-label { font-size: 13px; color: var(--text-secondary); }
.partners-hero-amount-row { align-items: baseline; gap: 14px; margin-top: 8px; flex-wrap: wrap; }
.partners-balance { font-family: var(--font-mono); font-feature-settings: "tnum" 1, "lnum" 1; font-weight: 800; font-size: 42px; line-height: 1; letter-spacing: -0.02em; color: var(--text-money); }
.partners-hero-accrued { font-size: 11px; }
.partners-balance-actions .btn { white-space: nowrap; }

/* Сетка 2 колонки → 1 на мобайле */
.partners-grid2 { display: grid; grid-template-columns: 1fr 1fr; gap: 16px; align-items: start; }

/* Карточка — тёмная, скруглённая (как .card эталона) */
.partners-card { background: var(--bg-container); border: 1px solid var(--border-default); border-radius: 14px; padding: 16px; }

/* Поле ссылки */
.partners-linkfield { display: flex; align-items: center; gap: 10px; background: var(--bg-app); border: 1px solid var(--border-default); border-radius: 9px; padding: 9px 12px; min-width: 0; }
.partners-link-ic { color: var(--text-tertiary); display: inline-flex; flex: 0 0 auto; }
.partners-link-value { flex: 1; min-width: 0; font-size: 13px; color: var(--text-primary); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.partners-link-value.placeholder { color: var(--text-placeholder); }
.partners-channel-name { width: 64px; font-size: 11px; flex: 0 0 auto; }

/* Строка списка приглашённых / истории (как .brow эталона) */
.partners-brow { display: flex; justify-content: space-between; align-items: baseline; gap: 12px; padding: 9px 0; border-bottom: 1px solid var(--border-default); }
.partners-brow:last-child { border-bottom: 0; }
.partners-brow > :first-child { min-width: 0; }
.partners-brow-main { font-weight: 600; font-size: 13px; }
.partners-brow-meta { font-size: 12px; flex-shrink: 0; }
.partners-tx-desc { font-size: 13px; }
.partners-tx-date { font-size: 10px; }
.partners-tx-amount { font-size: 13px; font-weight: 600; flex-shrink: 0; white-space: nowrap; }
.partners-tx-amount.pos { color: var(--text-success); }
.partners-tx-amount.neg { color: var(--text-tertiary); }
.partners-count { font-size: 11px; }
.partners-empty { text-align: center; padding: 22px 12px; border: 1px dashed var(--border-default); border-radius: 10px; color: var(--text-tertiary); font-size: 13px; }

/* Скелетон загрузки */
.skeleton-line { display: inline-block; border-radius: 6px; background: linear-gradient(90deg, var(--bg-container-2) 25%, var(--bg-neutral) 37%, var(--bg-container-2) 63%); background-size: 400% 100%; animation: partnersShimmer 1.3s ease-in-out infinite; }
@keyframes partnersShimmer { 0% { background-position: 100% 0; } 100% { background-position: 0 0; } }

/* Модалка «Продлить сервер» */
.server-extend-modal { min-width: 0; max-width: 460px; }
.server-extend-price { font-family: var(--font-mono); font-feature-settings: "tnum" 1, "lnum" 1; font-weight: 800; font-size: 38px; line-height: 1; color: var(--text-money); }
.server-extend-modal .money { color: var(--text-money); font-family: var(--font-mono); font-feature-settings: "tnum" 1, "lnum" 1; font-weight: 700; }
.server-extend-split { font-size: 13px; background: var(--bg-container-2); border: 1px solid var(--border-default); border-radius: 8px; padding: 10px 12px; }

@media (max-width: 640px) {
  .partners-screen { padding: 16px !important; }
  .partners-hero { padding: 18px; border-radius: 14px; }
  .partners-balance { font-size: 34px; }
  .partners-balance-actions { width: 100%; }
  .partners-balance-actions .btn { flex: 1; justify-content: center; min-height: 44px; }
  /* Сетка в одну колонку */
  .partners-grid2 { grid-template-columns: 1fr; }
  /* Поле канала: имя над значением, кнопка во всю ширину под палец */
  .partners-channel-name { width: auto; }
  .partners-linkfield .btn { min-height: 40px; }
}
