/*!*********************************************************************************************************************************************************************************************************************************************************************!*\
  !*** css ../../node_modules/next/dist/build/webpack/loaders/css-loader/src/index.js??ruleSet[1].rules[14].oneOf[12].use[2]!../../node_modules/next/dist/build/webpack/loaders/postcss-loader/src/index.js??ruleSet[1].rules[14].oneOf[12].use[3]!./app/globals.css ***!
  \*********************************************************************************************************************************************************************************************************************************************************************/
/* =============================================================================
   Sewing — mobile-first clean design (Шаг 13 / Pilot UX polish)
   ----------------------------------------------------------------------------
   Цели:
   - белый фон, светлые карточки, мягкие синие акценты;
   - крупные тач-цели и много воздуха для цеха;
   - mobile-first, но без поломки уже существующих desktop-экранов
     (admin, shopfloor, orders, earnings) — поэтому базовые токены и
     служебные классы (.card, .btn, .data-table, .status-badge,
     .form-row, .meta-line, .summary-card, .toolbar, .error-box и т.д.)
     сохранены; они только заиграли в новой палитре.
   ============================================================================= */

:root {
  /* Палитра ----------------------------------------------------------------- */
  --color-bg: #ffffff;
  --color-bg-soft: #f5f7fb;        /* фон страницы на мобиле */
  --color-bg-card: #ffffff;
  --color-bg-muted: #f1f4f9;       /* карточки/чипсы */
  --color-bg-tint: #eef3ff;        /* мягкий синий акцент-фон */

  --color-fg: #0f172a;             /* основной текст */
  --color-fg-strong: #0b1220;
  --color-fg-muted: #64748b;       /* вторичный текст */
  --color-fg-subtle: #94a3b8;

  --color-border: #e6ebf2;
  --color-border-strong: #d3dae4;

  --color-accent: #2563eb;         /* мягкий синий */
  --color-accent-hover: #1d4ed8;
  --color-accent-soft: #dbe6ff;
  --color-accent-fg: #1e3a8a;

  --color-ok: #16a34a;
  --color-ok-soft: #dcfce7;
  --color-ok-fg: #166534;

  --color-warn: #b45309;
  --color-warn-soft: #fef3c7;
  --color-warn-fg: #92400e;

  --color-danger: #dc2626;
  --color-danger-soft: #fee2e2;
  --color-danger-fg: #991b1b;

  /* Геометрия -------------------------------------------------------------- */
  --radius-sm: 8px;
  --radius-md: 12px;
  --radius-lg: 16px;
  --radius-xl: 22px;
  --radius-pill: 999px;

  --shadow-xs: 0 1px 2px rgba(15, 23, 42, 0.04);
  --shadow-sm: 0 1px 3px rgba(15, 23, 42, 0.06), 0 1px 2px rgba(15, 23, 42, 0.04);
  --shadow-md: 0 6px 18px rgba(15, 23, 42, 0.06);
  --shadow-lg: 0 12px 28px rgba(15, 23, 42, 0.10);

  /* Размер текста / UI ----------------------------------------------------- */
  --font-base: 16px;
  --tap-min: 48px;

  /* Алиасы для совместимости со старыми правилами -------------------------- */
  --fg: var(--color-fg);
  --fg-muted: var(--color-fg-muted);
  --bg: var(--color-bg-soft);
  --bg-card: var(--color-bg-card);
  --border: var(--color-border);
  --border-strong: var(--color-border-strong);
  --accent: var(--color-accent);
  --accent-hover: var(--color-accent-hover);
  --danger: var(--color-danger);
  --warn: var(--color-warn);
  --ok: var(--color-ok);

  --status-draft-bg: #e6ebf2;
  --status-draft-fg: #334155;
  --status-calculation-bg: #fef3c7;
  --status-calculation-fg: #92400e;
  --status-in-production-bg: #dbe6ff;
  --status-in-production-fg: #1d4ed8;
  --status-done-bg: #dcfce7;
  --status-done-fg: #166534;
  --status-cancelled-bg: #fee2e2;
  --status-cancelled-fg: #991b1b;
}

* { box-sizing: border-box; }

html, body {
  margin: 0;
  padding: 0;
  font-family: ui-sans-serif, system-ui, -apple-system, "Segoe UI", Roboto,
    "Helvetica Neue", Arial, sans-serif;
  font-size: var(--font-base);
  background: var(--color-bg);
  color: var(--color-fg);
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
  text-rendering: optimizeLegibility;
}

a { color: var(--color-accent); text-decoration: none; }
a:hover { color: var(--color-accent-hover); }

/* =============================================================
   App shell: header + main + (mobile) bottom nav
   ============================================================= */

.app-header {
  display: flex;
  align-items: center;
  gap: 1rem;
  padding: 0.65rem 1.25rem;
  background: #0f172a;            /* фирменный тёмно-синий — оставлен */
  color: #f8fafc;
  position: sticky;
  top: 0;
  z-index: 30;
  box-shadow: 0 1px 0 rgba(15, 23, 42, 0.08);
}
.app-header a { color: #cbd5e1; transition: color 0.15s; }
.app-header a:hover { color: #fff; }

.app-header__brand {
  font-weight: 700;
  font-size: 1.05rem;
  letter-spacing: 0.01em;
  display: flex;
  align-items: center;
  gap: 0.5rem;
}
.app-header__brand a { color: #fff; }

.app-header__nav {
  display: flex;
  gap: 0.25rem;
  flex-wrap: wrap;
  font-size: 0.92rem;
}
.app-header__nav a {
  padding: 0.4rem 0.7rem;
  border-radius: var(--radius-pill);
  transition: background 0.15s, color 0.15s;
}
.app-header__nav a:hover { background: rgba(255, 255, 255, 0.08); }

.app-header__user {
  margin-left: auto;
  display: flex;
  align-items: center;
  gap: 0.6rem;
  font-size: 0.85rem;
}
.app-header__user-name {
  font-weight: 600;
  color: #f1f5f9;
  max-width: 14rem;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.app-header__user-role {
  background: rgba(255, 255, 255, 0.08);
  color: #e2e8f0;
  padding: 0.15rem 0.55rem;
  border-radius: var(--radius-pill);
  font-size: 0.72rem;
  letter-spacing: 0.04em;
  text-transform: uppercase;
}
.app-header__login {
  background: var(--color-accent);
  color: #fff !important;
  padding: 0.4rem 0.85rem;
  border-radius: var(--radius-pill);
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
}
.app-header__login .app-icon { width: 14px; height: 14px; }
.app-header__logout {
  background: transparent;
  color: #cbd5e1;
  border: 1px solid rgba(255, 255, 255, 0.16);
  padding: 0.35rem 0.75rem;
  border-radius: var(--radius-pill);
  font-size: 0.82rem;
  cursor: pointer;
  transition: background 0.15s, color 0.15s, border-color 0.15s;
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
}
.app-header__logout .app-icon { width: 14px; height: 14px; }
.app-header__logout:hover {
  background: rgba(255, 255, 255, 0.08);
  color: #fff;
  border-color: rgba(255, 255, 255, 0.28);
}

.app-main {
  max-width: 1280px;
  margin: 0 auto;
  padding: 1.25rem 1.25rem calc(1.25rem + env(safe-area-inset-bottom, 0px));
  background: var(--color-bg);
  min-height: calc(100vh - 56px);
}

/* Нижняя мобильная навигация ----------------------------------------------- */
.mobile-nav {
  display: none;
  position: fixed;
  left: 0;
  right: 0;
  bottom: 0;
  z-index: 25;
  background: #ffffff;
  border-top: 1px solid var(--color-border);
  padding: 0.35rem 0.5rem calc(0.35rem + env(safe-area-inset-bottom, 0px));
  box-shadow: 0 -6px 18px rgba(15, 23, 42, 0.06);
}
.mobile-nav__list {
  display: grid;
  grid-template-columns: repeat(4, 1fr);
  gap: 0.25rem;
  margin: 0;
  padding: 0;
  list-style: none;
}
.mobile-nav__link {
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  gap: 0.15rem;
  padding: 0.4rem 0.25rem;
  border-radius: var(--radius-md);
  color: var(--color-fg-muted) !important;
  font-size: 0.7rem;
  font-weight: 600;
  letter-spacing: 0.01em;
  text-align: center;
  min-height: var(--tap-min);
}
.mobile-nav__icon {
  font-size: 1.15rem;
  line-height: 1;
}
.mobile-nav__link:hover { background: var(--color-bg-muted); }

/* =============================================================
   Заголовки страниц / общие блоки
   ============================================================= */
.page-header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  margin-bottom: 1.25rem;
  gap: 1rem;
  flex-wrap: wrap;
}
.page-header h1 { margin: 0; font-size: 1.5rem; letter-spacing: -0.01em; }

/* =============================================================
   Карточки
   ============================================================= */
.card {
  background: var(--color-bg-card);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  padding: 1.1rem 1.25rem;
  box-shadow: var(--shadow-xs);
}

/* Универсальная мягкая карточка — фон как у бледно-серой плашки. */
.section-card {
  background: var(--color-bg-muted);
  border: 1px solid transparent;
  border-radius: var(--radius-lg);
  padding: 1rem 1.1rem;
}
.section-card__title {
  margin: 0 0 0.75rem;
  font-size: 0.95rem;
  font-weight: 700;
  color: var(--color-fg);
  letter-spacing: -0.005em;
}
.section-card__hint {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  margin: 0 0 0.75rem;
  line-height: 1.45;
}

/* =============================================================
   Кнопки
   ============================================================= */
.btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 0.35rem;
  padding: 0.6rem 1.05rem;
  font-size: 0.95rem;
  font-weight: 600;
  border-radius: var(--radius-md);
  border: 1px solid var(--color-border-strong);
  background: #fff;
  color: var(--color-fg);
  cursor: pointer;
  text-decoration: none;
  transition: background 0.15s, border-color 0.15s, color 0.15s,
    box-shadow 0.15s, transform 0.05s;
  min-height: 44px;
  line-height: 1.2;
}
.btn:hover { background: #f8fafc; border-color: var(--color-fg-subtle); }
.btn:active { transform: translateY(1px); }
.btn-primary {
  background: var(--color-accent);
  color: #fff;
  border-color: var(--color-accent);
  box-shadow: 0 4px 14px -6px rgba(37, 99, 235, 0.55);
}
.btn-primary:hover {
  background: var(--color-accent-hover);
  border-color: var(--color-accent-hover);
  color: #fff;
}
.btn-danger {
  color: var(--color-danger);
  border-color: #fecaca;
  background: #fff;
}
.btn-danger:hover { background: #fef2f2; }
.btn-ghost {
  border: 1px solid transparent;
  background: transparent;
}
.btn-ghost:hover { background: var(--color-bg-muted); }
.btn[disabled],
.btn[aria-disabled="true"] {
  opacity: 0.5;
  cursor: not-allowed;
  box-shadow: none;
}

.btn-block { width: 100%; }
.btn-lg { min-height: 56px; font-size: 1.05rem; padding: 0.85rem 1.25rem; }

/* =============================================================
   Формы
   ============================================================= */
.form-row {
  display: grid;
  grid-template-columns: 180px 1fr;
  gap: 0.75rem;
  align-items: start;
  margin-bottom: 0.75rem;
}
.form-row label {
  font-weight: 600;
  padding-top: 0.5rem;
  color: var(--color-fg-muted);
}
.form-row .hint {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  margin-top: 0.3rem;
}

input[type="date"],
input[type="text"],
input[type="number"],
input[type="search"],
input[type="password"],
textarea,
select {
  width: 100%;
  padding: 0.65rem 0.85rem;
  border: 1px solid var(--color-border-strong);
  border-radius: var(--radius-md);
  font-size: 1rem;
  background: #fff;
  color: var(--color-fg);
  transition: border-color 0.15s, box-shadow 0.15s, background 0.15s;
}
input:focus,
textarea:focus,
select:focus {
  outline: none;
  border-color: var(--color-accent);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.18);
}
textarea { min-height: 80px; resize: vertical; }

/* =============================================================
   Таблицы (desktop / админ-экраны)
   ============================================================= */
table.data-table {
  width: 100%;
  border-collapse: collapse;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  overflow: hidden;
}
table.data-table th,
table.data-table td {
  padding: 0.65rem 0.85rem;
  text-align: left;
  border-bottom: 1px solid var(--color-border);
  font-size: 0.92rem;
}
table.data-table th {
  background: #f8fafc;
  font-weight: 600;
  color: var(--color-fg-muted);
  text-transform: uppercase;
  font-size: 0.72rem;
  letter-spacing: 0.05em;
}
table.data-table tr:last-child td { border-bottom: none; }
table.data-table td.num { text-align: right; font-feature-settings: "tnum"; font-variant-numeric: tabular-nums; }
table.data-table th.num { text-align: right; }

/* =============================================================
   Status badges + новые pill'ы
   ============================================================= */
.status-badge {
  display: inline-block;
  padding: 0.2rem 0.65rem;
  border-radius: var(--radius-pill);
  font-size: 0.72rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
}
.status-badge.draft       { background: var(--status-draft-bg); color: var(--status-draft-fg); }
.status-badge.calculation { background: var(--status-calculation-bg); color: var(--status-calculation-fg); }
.status-badge.in_production { background: var(--status-in-production-bg); color: var(--status-in-production-fg); }
.status-badge.done        { background: var(--status-done-bg); color: var(--status-done-fg); }
.status-badge.cancelled   { background: var(--status-cancelled-bg); color: var(--status-cancelled-fg); }
.status-badge.created     { background: var(--status-draft-bg); color: var(--status-draft-fg); }
.status-badge.in_progress { background: var(--status-in-production-bg); color: var(--status-in-production-fg); }
.status-badge.packed      { background: var(--status-done-bg); color: var(--status-done-fg); }
/* Cutting closure request statuses (ADR-0018). */
.status-badge.requested   { background: var(--status-in-production-bg); color: var(--status-in-production-fg); }
.status-badge.approved    { background: var(--status-done-bg); color: var(--status-done-fg); }
.status-badge.rejected    { background: var(--status-cancelled-bg); color: var(--status-cancelled-fg); }

/* Универсальный pill (новые экраны). */
.pill {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  padding: 0.25rem 0.7rem;
  border-radius: var(--radius-pill);
  font-size: 0.78rem;
  font-weight: 600;
  background: var(--color-bg-muted);
  color: var(--color-fg);
}
.pill--accent  { background: var(--color-accent-soft); color: var(--color-accent-fg); }
.pill--ok      { background: var(--color-ok-soft); color: var(--color-ok-fg); }
.pill--warn    { background: var(--color-warn-soft); color: var(--color-warn-fg); }
.pill--danger  { background: var(--color-danger-soft); color: var(--color-danger-fg); }
.pill--ghost   { background: transparent; color: var(--color-fg-muted); border: 1px solid var(--color-border); }

/* =============================================================
   Summary cards / counters / meta
   ============================================================= */
.summary-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(140px, 1fr));
  gap: 0.75rem;
  margin: 1rem 0 1.5rem;
}
.summary-card {
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  padding: 0.85rem 1rem;
}
.summary-card__label {
  font-size: 0.72rem;
  color: var(--color-fg-muted);
  text-transform: uppercase;
  letter-spacing: 0.05em;
  margin-bottom: 0.25rem;
}
.summary-card__value {
  font-size: 1.5rem;
  font-weight: 700;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.summary-card.delta .summary-card__value { color: var(--color-fg-muted); }
.summary-card.delta .summary-card__value.positive { color: var(--color-ok); }
.summary-card.delta .summary-card__value.negative { color: var(--color-danger); }

.toolbar {
  display: flex;
  gap: 0.5rem;
  align-items: center;
  flex-wrap: wrap;
  margin-bottom: 1rem;
}
.toolbar input,
.toolbar select { width: auto; }

.error-box {
  background: var(--color-danger-soft);
  color: var(--color-danger-fg);
  border: 1px solid #fecaca;
  border-radius: var(--radius-md);
  padding: 0.85rem 1rem;
  margin-bottom: 1rem;
}
.error-box__msg {
  font-weight: 600;
  font-size: 1.05rem;
  line-height: 1.35;
}
.error-box__hint {
  margin-top: 0.3rem;
  font-size: 0.92rem;
  font-weight: 500;
  line-height: 1.35;
  opacity: 0.85;
}
.error-box__rid {
  margin-top: 0.4rem;
  font-size: 0.8rem;
  color: #7f1d1d;
  opacity: 0.85;
  -webkit-user-select: all;
          user-select: all;
}
.error-box__rid code {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 0.8rem;
  background: rgba(127, 29, 29, 0.08);
  padding: 0 4px;
  border-radius: 3px;
}
.info-box {
  background: var(--color-ok-soft);
  color: var(--color-ok-fg);
  border: 1px solid #a7f3d0;
  border-radius: var(--radius-md);
  padding: 0.85rem 1rem;
  margin-bottom: 1rem;
}
.empty {
  color: var(--color-fg-muted);
  padding: 2rem;
  text-align: center;
}

/* =============================================================
   Size picker — крупные кнопки выбора размера (passport create)
   -------------------------------------------------------------
   Семантически радио-группа, визуально набор больших тач-кнопок.
   Реальные <input type="radio"> скрыты визуально, но остаются
   доступны клавиатурой и a11y-стеком.
   ============================================================= */
.size-picker {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(96px, 1fr));
  gap: 0.5rem;
}
.size-picker__option {
  position: relative;
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  gap: 0.15rem;
  min-height: 56px;
  padding: 0.55rem 0.6rem;
  background: #fff;
  border: 1.5px solid var(--color-border-strong);
  border-radius: var(--radius-md);
  color: var(--color-fg);
  cursor: pointer;
  text-align: center;
  -webkit-user-select: none;
          user-select: none;
  transition: background 0.15s, border-color 0.15s, color 0.15s,
    box-shadow 0.15s, transform 0.05s;
}
.size-picker__option:hover {
  border-color: var(--color-accent);
  background: var(--color-bg-tint);
}
.size-picker__option:active { transform: translateY(1px); }
.size-picker__option.is-active {
  background: var(--color-accent);
  border-color: var(--color-accent);
  color: #fff;
  box-shadow: 0 6px 16px -8px rgba(37, 99, 235, 0.55);
}
.size-picker__option.is-disabled {
  opacity: 0.45;
  cursor: not-allowed;
  background: var(--color-bg-muted);
  color: var(--color-fg-muted);
  border-color: var(--color-border);
  box-shadow: none;
}
.size-picker__option.is-disabled:hover {
  background: var(--color-bg-muted);
  border-color: var(--color-border);
}
.size-picker__input {
  position: absolute;
  opacity: 0;
  width: 1px;
  height: 1px;
  pointer-events: none;
}
.size-picker__input:focus-visible + .size-picker__code,
.size-picker__option:focus-within {
  outline: none;
}
.size-picker__option:focus-within {
  border-color: var(--color-accent);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.25);
}
.size-picker__code {
  font-size: 1.15rem;
  font-weight: 700;
  line-height: 1.1;
  letter-spacing: -0.01em;
}
.size-picker__meta {
  font-size: 0.72rem;
  font-weight: 600;
  color: var(--color-fg-muted);
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  letter-spacing: 0.02em;
}
.size-picker__option.is-active .size-picker__meta { color: rgba(255, 255, 255, 0.85); }
.size-picker__option.is-disabled .size-picker__meta { color: var(--color-fg-subtle); }

@media (max-width: 640px) {
  .size-picker { grid-template-columns: repeat(auto-fill, minmax(84px, 1fr)); }
  .size-picker__option { min-height: 60px; }
  .size-picker__code { font-size: 1.2rem; }
}

.size-grid {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(120px, 1fr));
  gap: 0.5rem;
}
.size-grid input { width: 100%; text-align: right; }
.size-grid label {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.3rem 0.55rem;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  background: #fff;
}
.size-grid label span { flex: 1 1; font-weight: 500; color: var(--color-fg-muted); }

.meta-line {
  color: var(--color-fg-muted);
  font-size: 0.9rem;
}
.meta-line strong { color: var(--color-fg); }
.meta-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
  gap: 0.5rem 1.5rem;
  margin-bottom: 1rem;
}
.actions-row {
  display: flex;
  gap: 0.5rem;
  flex-wrap: wrap;
  margin: 1rem 0;
}

/* =============================================================
   Admin overview (legacy блок — оставляем как есть, только палитра)
   ============================================================= */
.admin-overview { display: grid; gap: 1.5rem; }
.admin-overview__header {
  display: flex;
  align-items: flex-end;
  justify-content: space-between;
  gap: 1rem;
  flex-wrap: wrap;
}
.admin-overview__header h1 { margin: 0; font-size: 1.6rem; }
.admin-overview__counters {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(160px, 1fr));
  gap: 0.75rem;
}
.admin-overview__counter {
  background: var(--color-bg-card);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  padding: 0.95rem 1rem;
  text-align: center;
}
.admin-overview__counter-value {
  font-size: 1.9rem;
  font-weight: 700;
  color: var(--color-fg);
}
.admin-overview__counter-label {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  margin-top: 0.25rem;
}
.admin-overview .empty-hint {
  color: var(--color-fg-muted);
  font-style: italic;
  padding: 0.5rem 0;
}

/* =============================================================
   /admin/equipment — настройка операций оборудования (ADR-0017)
   ============================================================= */
.admin-equipment-form { display: grid; gap: 1rem; }
.admin-equipment-form__meta {
  display: flex;
  gap: 1rem;
  flex-wrap: wrap;
  align-items: baseline;
}
.meta-line--warn { color: var(--color-danger, #b91c1c); }
.success-box {
  background: #ecfdf5;
  border: 1px solid #a7f3d0;
  color: #065f46;
  border-radius: var(--radius-lg);
  padding: 0.6rem 0.85rem;
  font-size: 0.95rem;
}
/* Чекбокс-карточка с заголовком и подсказкой. Используется в форме
   выпуска паспорта (опция «Подать заявку на закрытие раскроя» для
   CUTTER_ASSISTANT, ADR-0018). Сделана крупной и mobile-friendly:
   тап-таргет ≥ 44 px, выделение активного состояния синей рамкой. */
.checkbox-card {
  display: flex;
  align-items: flex-start;
  gap: 0.65rem;
  padding: 0.75rem 0.9rem;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  background: var(--color-bg-card);
  cursor: pointer;
  min-height: 56px;
}
.checkbox-card.is-active {
  border-color: var(--color-accent, #2563eb);
  background: #eff6ff;
}
.checkbox-card input[type='checkbox'] {
  margin-top: 0.2rem;
  width: 20px;
  height: 20px;
  flex: 0 0 auto;
}
.checkbox-card__body {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
}
.checkbox-card__title {
  font-weight: 600;
  font-size: 1rem;
  color: var(--color-fg);
  line-height: 1.25;
}
.checkbox-card__hint {
  color: var(--color-fg-muted);
  font-size: 0.875rem;
  line-height: 1.35;
}
.admin-equipment-form__list {
  display: grid;
  gap: 0.5rem;
  list-style: none;
  margin: 0;
  padding: 0;
}
.admin-equipment-form__row {
  background: var(--color-bg-card);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  padding: 0.6rem 0.9rem;
}
.admin-equipment-form__row label {
  display: grid;
  grid-template-columns: auto 1fr auto;
  align-items: center;
  gap: 0.6rem;
  cursor: pointer;
}
.admin-equipment-form__op-name { font-weight: 600; color: var(--color-fg); }
.admin-equipment-form__op-meta { color: var(--color-fg-muted); font-size: 0.85rem; }

/* =============================================================
   /login — Auth screen (clean redesign)
   -------------------------------------------------------------
   Заменяет старую секцию login (auth-page / auth-card / auth-form):
   соответствующих правил здесь больше нет (см.
   `apps/web/components/auth/` и `docs/auth-design-cleanup-recon.md`).
   Все правила опираются только на existing UI-токены: палитра,
   радиусы и тени общие со всем приложением.
   ============================================================= */
.auth-screen {
  /* `100vh` — fallback для Safari < 15.4 / старых WebView, не знающих
     `dvh`; современные браузеры перезаписывают строкой ниже. */
  min-height: 100vh;
  min-height: 100dvh;
  display: flex;
  align-items: center;
  justify-content: center;
  padding: 1.5rem 1rem 2.25rem;
  background: var(--color-bg-soft);
}
.auth-screen__card {
  width: 100%;
  max-width: 440px;
  background: var(--color-bg-card);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  padding: 2rem 1.75rem 1.85rem;
  box-shadow: var(--shadow-md);
}
.auth-screen__header {
  display: flex;
  flex-direction: column;
  align-items: flex-start;
  gap: 0.6rem;
  margin-bottom: 1.5rem;
}
.auth-screen__brand-mark {
  width: 44px;
  height: 44px;
  border-radius: var(--radius-md);
  background: linear-gradient(135deg, var(--color-accent), #60a5fa);
  display: inline-flex;
  align-items: center;
  justify-content: center;
  color: #fff;
  font-weight: 700;
  font-size: 1.1rem;
  letter-spacing: 0.04em;
}
.auth-screen__title {
  margin: 0;
  font-size: 1.55rem;
  letter-spacing: -0.01em;
  color: var(--color-fg-strong);
}
.auth-screen__subtitle {
  margin: 0;
  color: var(--color-fg-muted);
  font-size: 0.95rem;
  line-height: 1.5;
}
.auth-screen__body {
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.auth-screen__form {
  display: flex;
  flex-direction: column;
  gap: 0.95rem;
}
.auth-screen__field {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
}
.auth-screen__label {
  font-size: 0.82rem;
  font-weight: 600;
  color: var(--color-fg-muted);
  letter-spacing: 0.01em;
}
.auth-screen__input {
  width: 100%;
  padding: 0.9rem 1rem;
  font: inherit;
  font-size: 1.05rem;
  color: var(--color-fg);
  background: var(--color-bg-card);
  border: 1px solid var(--color-border-strong);
  border-radius: var(--radius-md);
  transition: border-color 0.15s, box-shadow 0.15s;
}
.auth-screen__input::placeholder { color: var(--color-fg-subtle); }
.auth-screen__input:focus {
  outline: none;
  border-color: var(--color-accent);
  box-shadow: 0 0 0 4px var(--color-accent-soft);
}
.auth-screen__error {
  background: var(--color-danger-soft);
  color: var(--color-danger-fg);
  border: 1px solid #fecaca;
  border-radius: var(--radius-md);
  padding: 0.7rem 0.9rem;
  font-size: 0.95rem;
}
.auth-screen__submit {
  margin-top: 0.35rem;
  width: 100%;
  min-height: var(--tap-min);
  border: none;
  border-radius: var(--radius-md);
  background: var(--color-accent);
  color: #fff;
  font-size: 1.05rem;
  font-weight: 700;
  cursor: pointer;
  box-shadow: 0 6px 18px -8px rgba(37, 99, 235, 0.6);
  transition: background 0.15s, transform 0.05s;
}
.auth-screen__submit:hover { background: var(--color-accent-hover); }
.auth-screen__submit:active { transform: translateY(1px); }
.auth-screen__submit:disabled {
  opacity: 0.6;
  cursor: not-allowed;
  box-shadow: none;
}
.auth-screen__status {
  display: inline-flex;
  align-items: center;
  gap: 0.55rem;
  color: var(--color-fg-muted);
  font-size: 0.95rem;
}
.auth-screen__spinner {
  width: 14px;
  height: 14px;
  border-radius: var(--radius-pill);
  border: 2px solid var(--color-border-strong);
  border-top-color: var(--color-accent);
  animation: auth-screen-spin 0.7s linear infinite;
}
@keyframes auth-screen-spin {
  to { transform: rotate(360deg); }
}

/* =============================================================
   /work — рабочий мобильный кабинет
   ============================================================= */

.work {
  display: flex;
  flex-direction: column;
  gap: 1rem;
  position: relative;
}
.work-hint { color: var(--color-fg-muted); }

/* Mobile-clean режим швеи: убираем тяжёлый header, экран должен
   начинаться сразу с RoleHeaderCard. Добавляем безопасный отступ
   сверху под status bar (env(safe-area-inset-top)) и оставляем
   место под три-точечное меню действий справа. */
.work--seamstress {
  padding-top: calc(env(safe-area-inset-top, 0px));
}

/* Шапка-профиль (RoleHeaderCard) ------------------------------- */
.role-header {
  background: linear-gradient(135deg, #1e3a8a 0%, #1d4ed8 100%);
  color: #fff;
  border-radius: var(--radius-xl);
  padding: 1.1rem 1.2rem;
  box-shadow: var(--shadow-md);
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
  position: relative;
  overflow: hidden;
}
.role-header::after {
  content: "";
  position: absolute;
  right: -40px;
  top: -40px;
  width: 180px;
  height: 180px;
  border-radius: 50%;
  background: rgba(255, 255, 255, 0.06);
  pointer-events: none;
}
.role-header__top {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 0.75rem;
  position: relative;
  z-index: 1;
}
.role-header__brand {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.12em;
  color: rgba(255, 255, 255, 0.7);
  font-weight: 700;
}
.role-header__name {
  margin: 0.1rem 0 0;
  font-size: 1.35rem;
  font-weight: 700;
  letter-spacing: -0.01em;
  color: #fff;
}
.role-header__role {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  background: rgba(255, 255, 255, 0.16);
  color: #fff;
  padding: 0.3rem 0.7rem;
  border-radius: var(--radius-pill);
  font-size: 0.78rem;
  font-weight: 600;
  letter-spacing: 0.02em;
}
.role-header__shift {
  display: grid;
  gap: 0.5rem 1rem;
  grid-template-columns: repeat(auto-fit, minmax(140px, 1fr));
  position: relative;
  z-index: 1;
}
.role-header__shift > div {
  display: flex;
  flex-direction: column;
}
.role-header__shift-label {
  font-size: 0.7rem;
  letter-spacing: 0.1em;
  text-transform: uppercase;
  color: rgba(255, 255, 255, 0.7);
  margin-bottom: 0.1rem;
  font-weight: 600;
}
.role-header__shift-value {
  font-size: 1rem;
  font-weight: 600;
  color: #fff;
  line-height: 1.25;
}
.role-header__shift-meta {
  font-size: 0.78rem;
  color: rgba(255, 255, 255, 0.7);
}
.role-header__status {
  display: inline-flex;
  align-items: center;
  gap: 0.45rem;
  font-size: 0.85rem;
  color: rgba(255, 255, 255, 0.85);
}
.role-header__status-dot {
  width: 0.55rem;
  height: 0.55rem;
  border-radius: 50%;
  background: #94a3b8;
}
.role-header__status--active .role-header__status-dot {
  background: #34d399;
  box-shadow: 0 0 0 0 rgba(52, 211, 153, 0.6);
  animation: role-pulse 1.8s ease-out infinite;
}
@keyframes role-pulse {
  0% { box-shadow: 0 0 0 0 rgba(52, 211, 153, 0.55); }
  70% { box-shadow: 0 0 0 7px rgba(52, 211, 153, 0); }
  100% { box-shadow: 0 0 0 0 rgba(52, 211, 153, 0); }
}

/* Сетка крупных action-карточек -------------------------------- */
.action-grid {
  display: grid;
  gap: 0.75rem;
  grid-template-columns: repeat(2, 1fr);
}
.action-card {
  display: flex;
  flex-direction: column;
  align-items: flex-start;
  gap: 0.45rem;
  padding: 1rem 1.1rem;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  color: var(--color-fg);
  text-decoration: none;
  min-height: 110px;
  transition: border-color 0.15s, box-shadow 0.15s, transform 0.05s,
    background 0.15s;
  cursor: pointer;
  text-align: left;
  font: inherit;
}
.action-card:hover {
  border-color: var(--color-accent);
  box-shadow: 0 8px 24px -14px rgba(37, 99, 235, 0.45);
  background: #fff;
}
.action-card:active { transform: translateY(1px); }
.action-card--primary {
  background: linear-gradient(135deg, var(--color-accent) 0%, #3b82f6 100%);
  color: #fff;
  border-color: transparent;
  box-shadow: 0 12px 28px -16px rgba(37, 99, 235, 0.7);
}
.action-card--primary:hover { color: #fff; background: linear-gradient(135deg, #1d4ed8 0%, #2563eb 100%); }
.action-card--danger { color: var(--color-danger); }
.action-card--danger:hover { border-color: #fecaca; }

.action-card__icon {
  width: 36px;
  height: 36px;
  border-radius: 10px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  background: var(--color-bg-tint);
  color: var(--color-accent);
  font-size: 1.15rem;
}
.action-card--primary .action-card__icon {
  background: rgba(255, 255, 255, 0.16);
  color: #fff;
}
.action-card--danger .action-card__icon {
  background: var(--color-danger-soft);
  color: var(--color-danger);
}
.action-card__title {
  font-size: 1rem;
  font-weight: 700;
  line-height: 1.2;
}
.action-card__hint {
  font-size: 0.82rem;
  color: var(--color-fg-muted);
  line-height: 1.35;
}
.action-card--primary .action-card__hint { color: rgba(255, 255, 255, 0.85); }

/* Скан-блок ---------------------------------------------------- */
.scan-card {
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  padding: 1.1rem 1.15rem;
  box-shadow: var(--shadow-sm);
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
}
.scan-card__title {
  margin: 0;
  font-size: 1.1rem;
  font-weight: 700;
  display: inline-flex;
  align-items: center;
  letter-spacing: -0.005em;
  letter-spacing: -0.005em;
  color: var(--color-fg);
}
.scan-card__hint {
  margin: 0;
  font-size: 0.88rem;
  color: var(--color-fg-muted);
  line-height: 1.45;
}
/* Шапка карточки с местом под индикатор «есть задание» справа
   (помощник раскройщика: блок «Выпустить паспорт»). */
.scan-card__head-row {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 0.6rem;
}
/* Красный бейдж «N к выпуску» — есть завершённый раскрой, ожидающий
   выпуска паспортов. Пульсирующая точка привлекает внимание. */
.scan-card__task-badge {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  flex-shrink: 0;
  background: #fee2e2;
  color: #b91c1c;
  font-size: 0.78rem;
  font-weight: 700;
  padding: 0.25rem 0.6rem;
  border-radius: 999px;
  white-space: nowrap;
}
.scan-card__task-dot {
  width: 0.55rem;
  height: 0.55rem;
  border-radius: 50%;
  background: #dc2626;
  box-shadow: 0 0 0 0 rgba(220, 38, 38, 0.6);
  animation: scan-task-pulse 1.8s ease-out infinite;
}
@keyframes scan-task-pulse {
  0% { box-shadow: 0 0 0 0 rgba(220, 38, 38, 0.55); }
  70% { box-shadow: 0 0 0 6px rgba(220, 38, 38, 0); }
  100% { box-shadow: 0 0 0 0 rgba(220, 38, 38, 0); }
}
/* Подсветка всей карточки, когда есть задание: тонкая красная рамка. */
.scan-card--has-task {
  border-color: #fca5a5;
  box-shadow: 0 0 0 1px #fca5a5, var(--shadow-sm);
}
.scan-card__input {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
}
.scan-card__input-label {
  font-size: 0.78rem;
  font-weight: 700;
  text-transform: uppercase;
  letter-spacing: 0.06em;
  color: var(--color-fg-muted);
}
.scan-card__input input {
  font-size: 1.15rem;
  padding: 1rem 1.1rem;
  border-radius: var(--radius-md);
  background: var(--color-bg-soft);
  border: 1.5px dashed var(--color-border-strong);
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
}
.scan-card__input input:focus {
  background: #fff;
  border-style: solid;
  border-color: var(--color-accent);
}

/* Tabs «Получить крой / Сканировать» --------------------------- */
.work-tabs {
  display: grid;
  grid-template-columns: 1fr 1fr;
  gap: 0.5rem;
  background: var(--color-bg-muted);
  padding: 0.3rem;
  border-radius: var(--radius-pill);
}
.work-tab {
  appearance: none;
  border: none;
  background: transparent;
  padding: 0.7rem 0.9rem;
  border-radius: var(--radius-pill);
  font-weight: 600;
  font-size: 0.95rem;
  color: var(--color-fg-muted);
  cursor: pointer;
  min-height: 44px;
  transition: background 0.15s, color 0.15s, box-shadow 0.15s;
}
.work-tab:hover { color: var(--color-fg); }
.work-tab.is-active {
  background: #fff;
  color: var(--color-accent);
  box-shadow: var(--shadow-xs);
}

/* Контейнер «активной смены» (CUTTER + manager fallback на /work). */
/* `DefaultActivePanel` оборачивает табы и форму завершения смены      */
/* в `.work-active`, чтобы убрать inline `style={{ display: flex,      */
/* flexDirection: column, gap: 1rem }}`. См.                            */
/* `docs/design-cleanup-recon.md §7 Этап 3`.                             */
.work-active {
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.work-active__shift-end {
  display: flex;
  justify-content: flex-end;
}

/* Result cards (success/scan/error) ---------------------------- */
.result-card {
  border-radius: var(--radius-lg);
  border: 1px solid var(--color-border);
  background: var(--color-bg-soft);
  padding: 0.85rem 1rem;
  display: flex;
  flex-direction: column;
  gap: 0.55rem;
}
.result-card--issue { background: var(--color-ok-soft); border-color: #bbf7d0; }
.result-card--scan  { background: var(--color-bg-tint); border-color: #c5d8ff; }
.result-card__header {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 0.5rem;
  font-size: 0.95rem;
  font-weight: 700;
}
.result-card__row {
  display: flex;
  justify-content: space-between;
  gap: 0.6rem;
  font-size: 0.95rem;
  align-items: baseline;
  border-bottom: 1px dashed rgba(15, 23, 42, 0.08);
  padding: 0.25rem 0;
}
.result-card__row:last-child { border-bottom: none; }
.result-card__row span:first-child {
  color: var(--color-fg-muted);
  font-size: 0.85rem;
}
.result-card__row strong { font-weight: 700; }
/* Приглушённый «годных N» в строке «Количество» — заменяет inline */
/* style={{ color: var(--color-fg-muted), fontWeight: 500 }}.       */
.result-card__row-meta {
  color: var(--color-fg-muted);
  font-weight: 500;
}

/* Совместимость со старыми классами .work-* (на случай, если где-то ещё используются) */
.work-employee__label,
.work-section__title { /* keep */ }
.work-section__title { margin: 0 0 0.75rem; font-size: 1.1rem; font-weight: 700; }
.work-big-btn { min-height: 56px; font-size: 1.05rem; padding: 0.85rem 1.25rem; }

/* =============================================================
   Passport detail (mobile cards)
   ============================================================= */
.passport-detail {
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.passport-hero {
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  padding: 1.1rem 1.15rem;
  display: flex;
  gap: 1.25rem;
  align-items: center;
  flex-wrap: wrap;
  box-shadow: var(--shadow-sm);
}
.passport-hero__qr {
  flex: 0 0 auto;
  width: 200px;
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 0.4rem;
}
.passport-hero__qr img {
  width: 200px;
  height: 200px;
  image-rendering: pixelated;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  padding: 0.4rem;
}
.passport-hero__qr-code {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 0.78rem;
  color: var(--color-fg-muted);
  word-break: break-all;
  text-align: center;
}
.passport-hero__info {
  flex: 1 1 240px;
  min-width: 0;
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
}
.passport-hero__title {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  flex-wrap: wrap;
  margin: 0;
  font-size: 1.5rem;
  letter-spacing: -0.01em;
}
.passport-hero__sub {
  color: var(--color-fg-muted);
  font-size: 0.92rem;
}

.passport-fields {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(160px, 1fr));
  gap: 0.55rem 1rem;
}
.passport-field__label {
  font-size: 0.72rem;
  text-transform: uppercase;
  letter-spacing: 0.05em;
  color: var(--color-fg-muted);
  font-weight: 600;
  margin-bottom: 0.15rem;
}
.passport-field__value {
  font-size: 1rem;
  font-weight: 600;
  color: var(--color-fg);
}

.passport-section { display: flex; flex-direction: column; gap: 0.55rem; }
.passport-section__title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 700;
  letter-spacing: -0.005em;
  color: var(--color-fg);
}

.qc-summary {
  display: grid;
  grid-template-columns: repeat(3, 1fr) auto;
  gap: 0.75rem;
  align-items: center;
}
.qc-summary__cell {
  background: var(--color-bg-muted);
  border-radius: var(--radius-md);
  padding: 0.75rem 0.9rem;
}
.qc-summary__cell--defect { background: var(--color-danger-soft); }
.qc-summary__cell--good   { background: var(--color-ok-soft); }
.qc-summary__label {
  font-size: 0.72rem;
  letter-spacing: 0.05em;
  text-transform: uppercase;
  color: var(--color-fg-muted);
  font-weight: 600;
  margin-bottom: 0.2rem;
}
.qc-summary__value {
  font-size: 1.4rem;
  font-weight: 700;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--color-fg);
}
.qc-summary__cell--defect .qc-summary__value { color: var(--color-danger); }
.qc-summary__cell--good .qc-summary__value { color: var(--color-ok); }

/* =============================================================
   Mobile breakpoints
   ============================================================= */
@media (max-width: 900px) {
  .app-header__nav { display: none; }
  .mobile-nav { display: block; }
  .app-main { padding-bottom: calc(96px + env(safe-area-inset-bottom, 0px)); }
}

@media (max-width: 640px) {
  .app-header { padding: 0.55rem 0.85rem; gap: 0.6rem; }
  .app-header__brand { font-size: 1rem; }
  .app-header__user-name { max-width: 8rem; font-size: 0.82rem; }
  .app-header__user-role { display: none; }
  .app-main { padding-left: 0.85rem; padding-right: 0.85rem; padding-top: 0.85rem; }

  .page-header { gap: 0.5rem; margin-bottom: 0.85rem; }
  .page-header h1 { font-size: 1.3rem; }

  .form-row { grid-template-columns: 1fr; }
  .form-row label { padding-top: 0; }

  .action-grid { grid-template-columns: 1fr 1fr; }

  .passport-hero { padding: 1rem; gap: 1rem; }
  .passport-hero__qr,
  .passport-hero__qr img { width: 180px; height: auto; }
  .passport-hero__qr img { height: 180px; }
  .passport-hero__title { font-size: 1.25rem; }

  .qc-summary { grid-template-columns: repeat(3, 1fr); }
  .qc-summary > .actions-row,
  .qc-summary > a { grid-column: 1 / -1; justify-content: stretch; }

  table.data-table th,
  table.data-table td { font-size: 0.85rem; padding: 0.55rem 0.65rem; }
}

/* =============================================================
   /shopfloor (Шаг 10) — оставлено как было, только палитра
   подтянулась через переменные.
   ============================================================= */

/* =============================================================
   Modern visual refresh (UI v2) — единый design system layer.
   Эти классы добавляют новое, не ломая старые .card / .btn /
   .data-table / .summary-card / .status-badge / .pill /
   .action-card. См. docs/screens.md §«UI refresh».
   ============================================================= */

.app-icon {
  flex-shrink: 0;
  display: inline-block;
  vertical-align: -0.18em;
  color: currentColor;
}

/* Заголовок страницы с иконкой и подсказкой. Альтернатива .page-header
   для современных экранов (главная, дашборд, цеховые таблицы). */
.page-shell {
  display: flex;
  flex-direction: column;
  gap: 1.25rem;
}
.page-eyebrow {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  font-size: 0.74rem;
  font-weight: 700;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  color: var(--color-fg-muted);
}
.page-eyebrow .app-icon { width: 14px; height: 14px; }
.page-title {
  display: flex;
  align-items: center;
  gap: 0.55rem;
  margin: 0.2rem 0 0.35rem;
  font-size: 1.7rem;
  font-weight: 700;
  letter-spacing: -0.015em;
  color: var(--color-fg-strong);
  line-height: 1.15;
}
.page-title .app-icon {
  width: 28px;
  height: 28px;
  color: var(--color-accent);
}
.page-subtitle {
  margin: 0;
  font-size: 0.95rem;
  color: var(--color-fg-muted);
  line-height: 1.5;
  max-width: 64ch;
}
.section-header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.75rem;
  margin: 0 0 0.85rem;
  flex-wrap: wrap;
}
.section-header h2 {
  margin: 0;
  font-size: 1.1rem;
  letter-spacing: -0.005em;
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  color: var(--color-fg-strong);
}
.section-header h2 .app-icon {
  width: 18px;
  height: 18px;
  color: var(--color-fg-muted);
}
.section-header__hint {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
}

/* KPI / stat card — крупный показатель с иконкой и trend-подписью.
   Используется в production-dashboard и production-cost. */
.kpi-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(190px, 1fr));
  gap: 0.85rem;
}
.kpi-card {
  position: relative;
  display: flex;
  flex-direction: column;
  gap: 0.45rem;
  padding: 1rem 1.15rem 1.05rem;
  background: var(--color-bg-card);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  box-shadow: var(--shadow-xs);
  transition: border-color 0.15s, box-shadow 0.15s, transform 0.05s;
}
.kpi-card:hover {
  border-color: var(--color-border-strong);
  box-shadow: var(--shadow-md);
}
.kpi-card__head {
  display: flex;
  align-items: center;
  gap: 0.55rem;
  font-size: 0.78rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--color-fg-muted);
}
.kpi-card__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 32px;
  height: 32px;
  border-radius: 10px;
  background: var(--color-bg-tint);
  color: var(--color-accent);
}
.kpi-card__icon .app-icon { width: 18px; height: 18px; }
.kpi-card__value {
  font-size: 1.7rem;
  font-weight: 700;
  letter-spacing: -0.01em;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--color-fg-strong);
  line-height: 1.1;
}
.kpi-card__sub {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  line-height: 1.35;
}
.kpi-card--ok    { border-color: #bbf7d0; }
.kpi-card--ok    .kpi-card__icon { background: var(--color-ok-soft); color: var(--color-ok-fg); }
.kpi-card--ok    .kpi-card__value { color: var(--color-ok-fg); }
.kpi-card--warn  { border-color: #fde68a; }
.kpi-card--warn  .kpi-card__icon { background: var(--color-warn-soft); color: var(--color-warn-fg); }
.kpi-card--warn  .kpi-card__value { color: var(--color-warn-fg); }
.kpi-card--danger{ border-color: #fecaca; }
.kpi-card--danger .kpi-card__icon { background: var(--color-danger-soft); color: var(--color-danger-fg); }
.kpi-card--danger .kpi-card__value { color: var(--color-danger-fg); }
.kpi-card--accent{ border-color: #c5d8ff; }
.kpi-card--accent .kpi-card__icon { background: var(--color-accent-soft); color: var(--color-accent-fg); }
.kpi-card--accent .kpi-card__value { color: var(--color-accent-fg); }

/* Pipeline stage card (production dashboard §2). */
.stage-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(130px, 1fr));
  gap: 0.65rem;
}
.stage-card {
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
  padding: 0.8rem 0.9rem;
  background: var(--color-bg-card);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  transition: border-color 0.15s, background 0.15s;
}
.stage-card__head {
  display: flex;
  align-items: center;
  gap: 0.4rem;
  font-size: 0.78rem;
  font-weight: 600;
  color: var(--color-fg-muted);
  text-transform: uppercase;
  letter-spacing: 0.04em;
}
.stage-card__head .app-icon { width: 14px; height: 14px; }
.stage-card__value {
  font-size: 1.45rem;
  font-weight: 700;
  letter-spacing: -0.01em;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--color-fg-strong);
}
.stage-card__value-unit {
  font-size: 0.85rem;
  font-weight: 500;
  color: var(--color-fg-muted);
  margin-left: 0.2rem;
}
.stage-card__meta {
  font-size: 0.78rem;
  color: var(--color-fg-muted);
}
.stage-card--bottleneck {
  border-color: #fecaca;
  background: #fff7f7;
}
.stage-card--bottleneck .stage-card__head { color: var(--color-danger-fg); }
.stage-card--bottleneck .stage-card__value { color: var(--color-danger-fg); }
.stage-card--ghost {
  border-style: dashed;
  background: transparent;
}

/* Alert stack — единый язык severity для дашборда и менеджерских
   экранов. Совместим с .info-box / .error-box, не вытесняет их. */
.alert-stack {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.alert-row {
  display: flex;
  align-items: center;
  gap: 0.7rem;
  padding: 0.65rem 0.85rem;
  background: #fff;
  border: 1px solid var(--color-border);
  border-left: 3px solid var(--color-fg-subtle);
  border-radius: var(--radius-md);
  text-decoration: none;
  color: var(--color-fg);
  font-size: 0.95rem;
  line-height: 1.35;
  transition: background 0.15s, border-color 0.15s, transform 0.05s;
}
.alert-row:hover {
  background: var(--color-bg-soft);
}
.alert-row__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 28px;
  height: 28px;
  border-radius: 8px;
  background: var(--color-bg-muted);
  color: var(--color-fg-muted);
  flex-shrink: 0;
}
.alert-row__icon .app-icon { width: 16px; height: 16px; }
.alert-row__msg { flex-grow: 1; min-width: 0; }
.alert-row__value {
  font-weight: 700;
  white-space: nowrap;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.alert-row--info    { border-left-color: var(--color-accent); }
.alert-row--info    .alert-row__icon { background: var(--color-accent-soft); color: var(--color-accent-fg); }
.alert-row--warn    { border-left-color: #f59e0b; }
.alert-row--warn    .alert-row__icon { background: var(--color-warn-soft); color: var(--color-warn-fg); }
.alert-row--warn    .alert-row__value { color: var(--color-warn-fg); }
.alert-row--crit    { border-left-color: var(--color-danger); background: #fff7f7; }
.alert-row--crit    .alert-row__icon { background: var(--color-danger-soft); color: var(--color-danger-fg); }
.alert-row--crit    .alert-row__value { color: var(--color-danger-fg); }
.alert-row--success { border-left-color: var(--color-ok); }
.alert-row--success .alert-row__icon { background: var(--color-ok-soft); color: var(--color-ok-fg); }

/* Quick links — современные shortcut-карточки в дашборде. */
.quick-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
  gap: 0.6rem;
}
.quick-link {
  display: inline-flex;
  align-items: center;
  gap: 0.55rem;
  padding: 0.7rem 0.9rem;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  color: var(--color-fg);
  text-decoration: none;
  font-weight: 600;
  font-size: 0.95rem;
  transition: border-color 0.15s, background 0.15s, color 0.15s,
    box-shadow 0.15s;
  min-height: 48px;
}
.quick-link:hover {
  border-color: var(--color-accent);
  color: var(--color-accent);
  box-shadow: 0 6px 16px -12px rgba(37, 99, 235, 0.55);
}
.quick-link__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 32px;
  height: 32px;
  border-radius: 8px;
  background: var(--color-bg-tint);
  color: var(--color-accent);
  flex-shrink: 0;
}
.quick-link__icon .app-icon { width: 18px; height: 18px; }
.quick-link__label { flex-grow: 1; }
.quick-link__chev { color: var(--color-fg-subtle); }
.quick-link__chev .app-icon { width: 16px; height: 16px; }

/* action-card иконка как SVG (а не emoji). */
.action-card__icon .app-icon { width: 20px; height: 20px; }
.action-card__icon-svg { width: 20px; height: 20px; }
.action-card--primary .action-card__icon .app-icon { color: #fff; }

/* Header brand-mark — крупная цветная плашка с инициалом или
   иконкой. Используется в шапке и на login-карточке. */
.brand-mark {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 28px;
  height: 28px;
  border-radius: 8px;
  background: linear-gradient(135deg, var(--color-accent), #60a5fa);
  color: #fff;
  font-weight: 700;
  font-size: 0.9rem;
  flex-shrink: 0;
  box-shadow: 0 4px 10px -6px rgba(37, 99, 235, 0.6);
}
.brand-mark .app-icon { width: 16px; height: 16px; color: #fff; }

/* Header nav — иконка + текст. Cтильно деградирует на мобиле. */
.app-header__nav a {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
}
.app-header__nav .app-icon {
  width: 14px;
  height: 14px;
  color: rgba(255, 255, 255, 0.65);
}
.app-header__nav a:hover .app-icon { color: #fff; }
.mobile-nav__icon .app-icon { width: 22px; height: 22px; }

/* Tab/period toggle pill (production dashboard period switcher и др.) */
.toggle-group {
  display: inline-flex;
  align-items: center;
  gap: 0.2rem;
  padding: 0.25rem;
  background: var(--color-bg-muted);
  border-radius: var(--radius-pill);
}
.toggle-group__item {
  appearance: none;
  border: none;
  background: transparent;
  padding: 0.4rem 0.85rem;
  font-size: 0.85rem;
  font-weight: 600;
  color: var(--color-fg-muted);
  border-radius: var(--radius-pill);
  cursor: pointer;
  text-decoration: none;
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  min-height: 36px;
  transition: background 0.15s, color 0.15s;
}
.toggle-group__item:hover { color: var(--color-fg); }
.toggle-group__item.is-active {
  background: #fff;
  color: var(--color-accent);
  box-shadow: var(--shadow-xs);
}

/* Empty state — единый стиль для пустых разделов. */
.empty-state {
  display: flex;
  flex-direction: column;
  align-items: center;
  text-align: center;
  gap: 0.45rem;
  padding: 2rem 1rem;
  color: var(--color-fg-muted);
  background: var(--color-bg-soft);
  border: 1px dashed var(--color-border-strong);
  border-radius: var(--radius-lg);
}
.empty-state__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 44px;
  height: 44px;
  border-radius: 14px;
  background: var(--color-bg-muted);
  color: var(--color-fg-subtle);
}
.empty-state__icon .app-icon { width: 22px; height: 22px; }
.empty-state__title {
  font-size: 1rem;
  font-weight: 700;
  color: var(--color-fg);
}
.empty-state__hint {
  font-size: 0.9rem;
  color: var(--color-fg-muted);
  max-width: 40ch;
}

/* Improved data-table look — мягкий hover, аккуратная типографика.
   Применяется поверх существующего `.data-table` без слома схемы. */
table.data-table tbody tr { transition: background 0.12s; }
table.data-table tbody tr:hover td,
table.data-table tbody tr:hover th { background: var(--color-bg-soft); }
table.data-table th .app-icon {
  width: 14px;
  height: 14px;
  vertical-align: -2px;
  margin-right: 0.25rem;
  color: var(--color-fg-subtle);
}
table.data-table td code {
  background: var(--color-bg-muted);
  border-radius: 4px;
  padding: 0 0.3rem;
  font-size: 0.8rem;
  color: var(--color-fg-muted);
}

/* Toolbar пилюля для filter-блоков (production-cost, earnings). */
.filter-card {
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  padding: 1rem 1.15rem;
  margin-bottom: 1rem;
  display: flex;
  flex-wrap: wrap;
  gap: 0.85rem;
  align-items: flex-end;
}
.filter-card__field {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
  min-width: 0;
}
.filter-card__field label {
  font-size: 0.75rem;
  font-weight: 700;
  color: var(--color-fg-muted);
  text-transform: uppercase;
  letter-spacing: 0.04em;
}
.filter-card__field input,
.filter-card__field select {
  min-width: 160px;
}
.filter-card__actions {
  display: inline-flex;
  gap: 0.45rem;
  margin-left: auto;
}

/* =============================================================
   Detail page shell (UI v2.1) — единый стандарт detail-страниц.
   -------------------------------------------------------------
   Используется на /admin/employees/[id], /admin/operations/[id],
   /admin/warehouses/[id], /admin/equipment/[id], /passports/[id].
   Поверх .page-shell даёт унифицированную шапку (eyebrow / title /
   subtitle / status / actions) и набор мелких primitives для
   повторяющихся блоков внутри карточек: data-list (label/value),
   detail-form (form-row + actions-row + сообщения), inline-table,
   danger-zone.
   ============================================================= */

.detail-header {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 1.25rem;
  flex-wrap: wrap;
}
.detail-header__main { min-width: 0; flex: 1 1 320px; }
.detail-header__back {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  text-decoration: none;
  margin-bottom: 0.4rem;
}
.detail-header__back:hover { color: var(--color-accent); }
.detail-header__back .app-icon { width: 14px; height: 14px; }
.detail-header__meta {
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem 0.75rem;
  align-items: center;
  margin-top: 0.55rem;
  color: var(--color-fg-muted);
  font-size: 0.9rem;
}
.detail-header__meta code {
  background: var(--color-bg-muted);
  border-radius: 4px;
  padding: 0 0.35rem;
  font-size: 0.82rem;
}
.detail-header__badges {
  display: inline-flex;
  flex-wrap: wrap;
  gap: 0.4rem;
  margin-top: 0.45rem;
}
.detail-header__aside {
  display: flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 0.55rem;
  flex-shrink: 0;
}
.detail-header__actions {
  display: inline-flex;
  flex-wrap: wrap;
  gap: 0.45rem;
  justify-content: flex-end;
}

/* Универсальная сетка label / value для реквизитов внутри секции. */
.data-list {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
  gap: 0.85rem 1.4rem;
}
.data-list__item {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  min-width: 0;
}
.data-list__label {
  font-size: 0.72rem;
  font-weight: 700;
  letter-spacing: 0.06em;
  text-transform: uppercase;
  color: var(--color-fg-muted);
}
.data-list__value {
  font-size: 0.98rem;
  font-weight: 600;
  color: var(--color-fg);
  line-height: 1.3;
  word-break: break-word;
}
.data-list__value code {
  background: var(--color-bg-muted);
  border-radius: 4px;
  padding: 0 0.35rem;
  font-size: 0.85rem;
  color: var(--color-fg);
}
.data-list__value--muted { color: var(--color-fg-muted); font-weight: 500; }

/* Унифицированная форма detail-страницы. Поверх существующих
   .form-row / .btn — даёт собственный gap и зону кнопок «Сохранить /
   Отменить» снизу. Сообщения success/error лежат под формой. */
.detail-form {
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.detail-form__grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(220px, 1fr));
  gap: 0.85rem 1.1rem;
}
.detail-form__field {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
  min-width: 0;
}
.detail-form__field > label,
.detail-form__field > .detail-form__label {
  font-size: 0.78rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--color-fg-muted);
}
.detail-form__field--inline {
  flex-direction: row;
  align-items: center;
  gap: 0.55rem;
  min-height: 44px;
}
.detail-form__field--inline > label {
  text-transform: none;
  letter-spacing: 0;
  font-weight: 600;
  color: var(--color-fg);
  font-size: 0.95rem;
}
.detail-form__field--inline input[type='checkbox'] {
  width: 20px;
  height: 20px;
  margin: 0;
  accent-color: var(--color-accent);
}
.detail-form__hint {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  line-height: 1.4;
}
.detail-form__row {
  display: flex;
  flex-wrap: wrap;
  gap: 0.6rem;
  align-items: center;
}
.detail-form__actions {
  display: flex;
  flex-wrap: wrap;
  gap: 0.5rem;
  align-items: center;
  padding-top: 0.4rem;
  border-top: 1px dashed var(--color-border);
}
.detail-form__success,
.detail-form__error {
  display: flex;
  align-items: flex-start;
  gap: 0.55rem;
  border-radius: var(--radius-md);
  padding: 0.7rem 0.85rem;
  font-size: 0.92rem;
  line-height: 1.4;
}
.detail-form__success {
  background: var(--color-ok-soft);
  color: var(--color-ok-fg);
  border: 1px solid #a7f3d0;
}
.detail-form__error {
  background: var(--color-danger-soft);
  color: var(--color-danger-fg);
  border: 1px solid #fecaca;
}
.detail-form__error .app-icon,
.detail-form__success .app-icon {
  flex-shrink: 0;
  margin-top: 2px;
}
.detail-form__error-rid {
  display: block;
  font-size: 0.78rem;
  margin-top: 0.25rem;
  opacity: 0.8;
}
.detail-form__error-rid code {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  background: rgba(127, 29, 29, 0.08);
  padding: 0 4px;
  border-radius: 3px;
}

/* Inline-table — таблица внутри detail-секции. Тонкая обводка,
   внутренний padding согласован с .data-table, hover из v2. */
.inline-table-wrap {
  margin: 0;
  overflow-x: auto;
  border-radius: var(--radius-lg);
  border: 1px solid var(--color-border);
  background: #fff;
}
.inline-table-wrap > table.data-table { border: none; border-radius: 0; }

/* Полоска действий рядом со строкой таблицы (печать QR, отвязать). */
.table-actions {
  display: inline-flex;
  gap: 0.4rem;
  flex-wrap: wrap;
  justify-content: flex-end;
}

/* Danger-zone — выделенный блок для опасных действий (отвязка ячейки,
   архив, и т.п.). Не используется как самостоятельная секция, кладётся
   внутрь существующих .card / .section-card. */
.danger-zone {
  margin-top: 1rem;
  padding: 0.85rem 1rem;
  background: #fff7f7;
  border: 1px dashed #fecaca;
  border-radius: var(--radius-md);
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.danger-zone__title {
  font-size: 0.85rem;
  font-weight: 700;
  color: var(--color-danger-fg);
  letter-spacing: 0.02em;
}
.danger-zone__hint {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  margin: 0;
  line-height: 1.4;
}

/* Pricing-mode highlight (operations detail). Делает FIXED / BY_SIZE /
   SALARY_ONLY мгновенно считываемым в шапке. */
.pricing-mode {
  display: inline-flex;
  align-items: center;
  gap: 0.45rem;
  padding: 0.3rem 0.7rem;
  border-radius: var(--radius-pill);
  font-size: 0.8rem;
  font-weight: 700;
  background: var(--color-bg-muted);
  color: var(--color-fg);
  letter-spacing: 0.02em;
}
.pricing-mode .app-icon { width: 14px; height: 14px; }
.pricing-mode--fixed       { background: var(--color-accent-soft); color: var(--color-accent-fg); }
.pricing-mode--by-size     { background: #fef3c7; color: var(--color-warn-fg); }
.pricing-mode--salary-only { background: var(--color-bg-muted); color: var(--color-fg-muted); }

/* Rate table for operations BY_SIZE — крупные ячейки, выровненные
   столбцы, акцентные плейсхолдеры. */
.rate-table-grid {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
  gap: 0.5rem;
}
.rate-cell {
  display: flex;
  flex-direction: column;
  gap: 0.25rem;
  padding: 0.55rem 0.65rem;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  transition: border-color 0.15s, box-shadow 0.15s;
}
.rate-cell:focus-within {
  border-color: var(--color-accent);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.18);
}
.rate-cell__size {
  font-size: 0.78rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--color-fg-muted);
}
.rate-cell__input {
  border: none;
  background: transparent;
  padding: 0;
  font-size: 1.05rem;
  font-weight: 700;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--color-fg);
  text-align: right;
}
.rate-cell__input:focus { outline: none; box-shadow: none; }

/* Компактная сетка карточек «Норма времени по размерам» —
   визуально парная к .rate-table-grid выше, но с двумя инпутами
   (минуты + секунды) в одной карточке. Не таблица, чтобы блок
   не растягивал страницу вниз (см. карточку операции BY_SIZE). */
.operation-time-size-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(110px, 1fr));
  gap: 0.5rem;
}
.operation-time-size-card {
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
  padding: 0.55rem 0.65rem;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  transition: border-color 0.15s, box-shadow 0.15s;
}
.operation-time-size-card:focus-within {
  border-color: var(--color-accent);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.18);
}
.operation-time-size-card__size {
  font-size: 0.78rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--color-fg-muted);
}
.operation-time-size-card__inputs {
  display: grid;
  grid-template-columns: 1fr 1fr;
  gap: 0.35rem;
}
.operation-time-size-card__inputs input {
  width: 100%;
  height: 36px;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-sm);
  padding: 0 0.45rem;
  font-size: 0.95rem;
  font-weight: 600;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  text-align: center;
  color: var(--color-fg);
  background: var(--color-bg-muted);
}
.operation-time-size-card__inputs input::placeholder {
  font-weight: 400;
  color: var(--color-fg-muted);
}
.operation-time-size-card__inputs input:focus {
  outline: none;
  border-color: var(--color-accent);
  background: #fff;
  box-shadow: 0 0 0 2px rgba(37, 99, 235, 0.12);
}
.operation-time-bulk-row {
  display: flex;
  flex-wrap: wrap;
  gap: 0.5rem;
  align-items: center;
  margin-bottom: 0.6rem;
}

/* Унифицированный список «attachable items» (operations checklist
   на equipment detail). Используется поверх существующих
   .admin-equipment-form__row, но с подсветкой выбранного. */
.option-list {
  display: grid;
  gap: 0.5rem;
  list-style: none;
  margin: 0;
  padding: 0;
}
.option-list__row {
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  padding: 0.6rem 0.9rem;
  transition: border-color 0.15s, background 0.15s;
}
.option-list__row.is-active {
  border-color: var(--color-accent);
  background: var(--color-bg-tint);
}
.option-list__row label {
  display: grid;
  grid-template-columns: auto 1fr auto;
  align-items: center;
  gap: 0.65rem;
  cursor: pointer;
  margin: 0;
  font-weight: 500;
}
.option-list__row label input[type='checkbox'] {
  width: 20px;
  height: 20px;
  margin: 0;
  accent-color: var(--color-accent);
}
.option-list__row-name {
  font-weight: 600;
  color: var(--color-fg);
  font-size: 0.95rem;
}
.option-list__row-meta {
  color: var(--color-fg-muted);
  font-size: 0.82rem;
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
}

/* Hero passport — уточняем верхний блок, оставляя совместимым. */
.passport-hero__badges {
  display: inline-flex;
  flex-wrap: wrap;
  gap: 0.4rem;
  margin-top: 0.15rem;
}

/* ==== /Detail page shell ==== */

/* ==== /Modern visual refresh ==== */

.shopfloor { display: flex; flex-direction: column; gap: 1rem; }
.shopfloor-board { display: flex; flex-direction: column; gap: 1rem; }

.shopfloor-toolbar {
  display: flex;
  flex-wrap: wrap;
  gap: 1rem;
  align-items: center;
  justify-content: space-between;
}
.shopfloor-title { display: flex; align-items: baseline; gap: 1rem; flex-wrap: wrap; }
.shopfloor-title h1 {
  margin: 0;
  font-size: 1.75rem;
  display: inline-flex;
  align-items: center;
  letter-spacing: -0.015em;
}
.shopfloor-title h1 .app-icon { color: var(--color-accent); }
.shopfloor-scope {
  color: var(--color-fg-muted);
  font-size: 1rem;
  font-weight: 500;
}
.shopfloor-controls {
  display: flex;
  gap: 0.75rem;
  align-items: center;
  flex-wrap: wrap;
}
.shopfloor-control {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  font-size: 0.8rem;
  color: var(--color-fg-muted);
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: 0.04em;
}
.shopfloor-control select { min-width: 220px; }
.shopfloor-toggle {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  font-size: 0.9rem;
  color: var(--color-fg-muted);
  -webkit-user-select: none;
          user-select: none;
}
.shopfloor-status {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.shopfloor-status__dot {
  width: 0.55rem;
  height: 0.55rem;
  border-radius: 50%;
  background: #94a3b8;
}
.shopfloor-status--on .shopfloor-status__dot {
  background: var(--color-ok);
  box-shadow: 0 0 0 0 rgba(22, 163, 74, 0.5);
  animation: shopfloor-pulse 1.6s ease-out infinite;
}
.shopfloor-poll-error { margin: 0; }

@keyframes shopfloor-pulse {
  0% { box-shadow: 0 0 0 0 rgba(22, 163, 74, 0.5); }
  70% { box-shadow: 0 0 0 8px rgba(22, 163, 74, 0); }
  100% { box-shadow: 0 0 0 0 rgba(22, 163, 74, 0); }
}

.shopfloor-summary {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
  gap: 0.75rem;
}
.shopfloor-summary__card {
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  padding: 0.85rem 1rem;
  transition: background 0.15s, border-color 0.15s;
}
.shopfloor-summary__card--defect {
  border-color: #fecaca;
  background: #fff7f7;
}
.shopfloor-summary__label {
  font-size: 0.75rem;
  color: var(--color-fg-muted);
  text-transform: uppercase;
  letter-spacing: 0.05em;
  margin-bottom: 0.25rem;
}
.shopfloor-summary__value {
  font-size: 2rem;
  font-weight: 700;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--color-fg);
}
.shopfloor-summary__value.is-zero { color: #cbd5e1; }
.shopfloor-summary__card--defect .shopfloor-summary__value { color: var(--color-danger); }
.shopfloor-summary__card--defect .shopfloor-summary__value.is-zero { color: #fecaca; }

.shopfloor-table-wrap { overflow-x: auto; }
.shopfloor-table {
  width: 100%;
  border-collapse: separate;
  border-spacing: 0;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  overflow: hidden;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.shopfloor-table th,
.shopfloor-table td {
  padding: 0.85rem 1rem;
  border-bottom: 1px solid var(--color-border);
  text-align: center;
  font-size: 1.7rem;
  font-weight: 700;
  color: var(--color-fg);
  background: #fff;
  transition: background 0.2s, color 0.2s, box-shadow 0.2s;
}
.shopfloor-table thead th {
  background: #0f172a;
  color: #e2e8f0;
  font-size: 0.85rem;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: 0.06em;
  padding: 0.55rem 1rem;
}
.shopfloor-table tbody tr:last-child td,
.shopfloor-table tbody tr:last-child th { border-bottom: none; }
.shopfloor-table tbody tr:nth-child(even) td,
.shopfloor-table tbody tr:nth-child(even) th { background: #f8fafc; }
.shopfloor-table__size {
  font-size: 1.4rem;
  background: #f1f5f9 !important;
  color: var(--color-fg-muted);
  text-align: left;
  width: 1%;
  white-space: nowrap;
}
.shopfloor-cell.is-zero { color: #cbd5e1; }
.shopfloor-cell--defect { color: var(--color-danger); }
.shopfloor-cell--defect.is-zero { color: #fecaca; }

.shopfloor-cell--flash-up,
.shopfloor-summary__card.shopfloor-cell--flash-up {
  animation: shopfloor-flash-up 1100ms ease-out;
}
.shopfloor-cell--flash-down,
.shopfloor-summary__card.shopfloor-cell--flash-down {
  animation: shopfloor-flash-down 1100ms ease-out;
}

@keyframes shopfloor-flash-up {
  0%   { background: #bbf7d0; box-shadow: inset 0 0 0 4px #16a34a; }
  60%  { background: #dcfce7; box-shadow: inset 0 0 0 2px #86efac; }
  100% { background: inherit; box-shadow: none; }
}
@keyframes shopfloor-flash-down {
  0%   { background: #fecaca; box-shadow: inset 0 0 0 4px #dc2626; }
  60%  { background: #fee2e2; box-shadow: inset 0 0 0 2px #fca5a5; }
  100% { background: inherit; box-shadow: none; }
}

.shopfloor-empty {
  text-align: center;
  color: var(--color-fg-muted);
  padding: 3rem 1rem;
  font-size: 1.05rem;
}

@media (max-width: 900px) {
  .shopfloor-table th,
  .shopfloor-table td { font-size: 1.3rem; padding: 0.65rem 0.6rem; }
  .shopfloor-table__size { font-size: 1.1rem; }
  .shopfloor-summary__value { font-size: 1.6rem; }
}
@media (min-width: 1500px) {
  .shopfloor-table th,
  .shopfloor-table td { font-size: 2.1rem; padding: 1rem 1.25rem; }
  .shopfloor-table__size { font-size: 1.6rem; }
  .shopfloor-summary__value { font-size: 2.4rem; }
}

/* =============================================================
   QR-сканер на /work (модальное окно поверх формы)
   ============================================================= */
.qr-modal {
  position: fixed;
  inset: 0;
  z-index: 100;
  background: rgba(15, 23, 42, 0.55);
  display: flex;
  align-items: flex-start;
  justify-content: center;
  padding: 1rem;
  overflow-y: auto;
  overscroll-behavior: contain;
}
.qr-modal__card {
  background: #fff;
  width: min(100%, 520px);
  margin: auto;
  border-radius: var(--radius-xl);
  padding: 1rem 1rem 1.1rem;
  display: flex;
  flex-direction: column;
  gap: 0.75rem;
  box-shadow: 0 20px 50px -10px rgba(15, 23, 42, 0.35);
}
.qr-modal__header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.5rem;
}
.qr-modal__title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 700;
  color: var(--color-fg);
}
.qr-modal__close {
  appearance: none;
  background: transparent;
  border: none;
  font-size: 1.6rem;
  line-height: 1;
  width: 40px;
  height: 40px;
  border-radius: var(--radius-md);
  cursor: pointer;
  color: var(--color-fg-muted);
}
.qr-modal__close:hover { background: var(--color-bg-muted); color: var(--color-fg); }
.qr-modal__hint {
  margin: 0;
  font-size: 0.9rem;
  color: var(--color-fg-muted);
  line-height: 1.4;
}
.qr-modal__hint--mono {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 0.78rem;
  color: var(--color-fg-subtle);
  word-break: break-word;
}
.qr-modal__viewport {
  width: 100%;
  aspect-ratio: 1 / 1;
  background: #0f172a;
  border-radius: var(--radius-lg);
  overflow: hidden;
  position: relative;
}
.qr-modal__viewport :global(video),
.qr-modal__viewport video {
  width: 100% !important;
  height: 100% !important;
  object-fit: cover;
}
.qr-modal__error {
  background: var(--color-danger-soft, #fef2f2);
  border: 1px solid #fecaca;
  border-radius: var(--radius-md);
  padding: 0.75rem 0.85rem;
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
}
.qr-modal__error-title {
  margin: 0;
  font-weight: 700;
  color: var(--color-danger, #b91c1c);
}
.qr-modal__cancel { flex: 0 0 auto; min-height: 52px; }
.scan-card__camera { min-height: 48px; font-weight: 600; }

/* Упрощённый /work для роли SEAMSTRESS ------------------------- */
.seamstress-work {
  display: flex;
  flex-direction: column;
  gap: 1.1rem;
}
/* Постоянный баннер «Очередь выдачи кроя» на /work для швеи
   (см. `cut-issue-rule-banner.tsx`). Mobile-first карточка с
   крупным размером и списком ячеек — UI должен быть читаем
   через всю мастерскую без приближения. */
.cut-issue-banner {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
}
.cut-issue-banner__card {
  background: var(--color-bg);
  border: 2px solid var(--color-accent);
  border-radius: var(--radius-xl);
  padding: 0.9rem 1rem;
  display: flex;
  flex-direction: column;
  gap: 0.45rem;
}
.cut-issue-banner__head {
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem 0.8rem;
  align-items: baseline;
  border-bottom: 1px solid var(--color-border, rgba(15, 23, 42, 0.1));
  padding-bottom: 0.35rem;
}
.cut-issue-banner__order {
  font-weight: 700;
  font-size: 0.95rem;
}
.cut-issue-banner__product {
  color: var(--color-fg-muted);
  font-size: 0.85rem;
}
.cut-issue-banner__row {
  display: flex;
  justify-content: space-between;
  align-items: baseline;
  gap: 0.6rem;
}
.cut-issue-banner__row--cells {
  flex-direction: column;
  align-items: stretch;
  gap: 0.25rem;
}
.cut-issue-banner__label {
  color: var(--color-fg-muted);
  font-size: 0.85rem;
  text-transform: uppercase;
  letter-spacing: 0.04em;
}
.cut-issue-banner__size {
  font-size: 1.6rem;
  font-weight: 800;
  letter-spacing: 0.02em;
}
.cut-issue-banner__remaining {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  font-weight: 600;
}
.cut-issue-banner__cells {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem;
}
.cut-issue-banner__cell {
  background: var(--color-bg-muted, rgba(15, 23, 42, 0.05));
  border-radius: var(--radius-md, 8px);
  padding: 0.25rem 0.6rem;
  display: inline-flex;
  align-items: baseline;
  gap: 0.4rem;
}
.cut-issue-banner__cell-code {
  font-weight: 700;
}
.cut-issue-banner__cell-qty {
  color: var(--color-fg-muted);
  font-size: 0.85rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.cut-issue-banner__cells-empty {
  color: var(--color-fg-muted);
  font-style: italic;
  font-size: 0.9rem;
}
/* Модалка «не тот размер» (см. `seamstress-active-panel.tsx ::
   WrongSizeModal`). Поднимается до PassportConfirmModal, поэтому
   z-index должен быть >= 100, как у `.qr-modal`. */
.modal-backdrop {
  position: fixed;
  inset: 0;
  z-index: 100;
  background: rgba(15, 23, 42, 0.55);
  display: flex;
  align-items: flex-start;
  justify-content: center;
  padding: 1rem;
  overflow-y: auto;
  overscroll-behavior: contain;
}
.modal {
  background: #fff;
  width: min(100%, 480px);
  margin: auto;
  border-radius: var(--radius-xl);
  padding: 1.1rem 1.2rem 1.2rem;
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  box-shadow: 0 20px 50px -10px rgba(15, 23, 42, 0.35);
}
.modal__title {
  margin: 0 0 0.2rem;
  font-size: 1.2rem;
  font-weight: 800;
}
.modal__text {
  margin: 0;
  font-size: 1rem;
  line-height: 1.4;
}
.modal__text--muted {
  color: var(--color-fg-muted);
}
.modal__cells {
  background: var(--color-bg-muted, rgba(15, 23, 42, 0.05));
  border-radius: var(--radius-md, 8px);
  padding: 0.5rem 0.75rem;
}
.modal__cells-label {
  font-size: 0.85rem;
  color: var(--color-fg-muted);
  text-transform: uppercase;
  letter-spacing: 0.04em;
  margin-bottom: 0.25rem;
}
.modal__cells-list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
}
.modal__actions {
  flex: 0 0 auto;
  display: flex;
  justify-content: flex-end;
  margin-top: 0.5rem;
}

/* OperationSwitcher: chip над активной панелью /work + модалка выбора.
   Логика — `apps/web/app/work/operation-switcher.tsx`. */
.operation-switcher__chip {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.75rem;
  width: 100%;
  padding: 0.55rem 0.9rem;
  background: var(--color-bg-muted, rgba(15, 23, 42, 0.05));
  border: 1px solid rgba(15, 23, 42, 0.08);
  border-radius: var(--radius-md, 10px);
  font: inherit;
  text-align: left;
  cursor: pointer;
}
.operation-switcher__chip:hover {
  background: rgba(15, 23, 42, 0.08);
}
.operation-switcher__chip-label {
  font-size: 0.95rem;
  color: var(--color-fg-muted);
}
.operation-switcher__chip-label strong {
  color: var(--color-fg);
  font-weight: 700;
  margin-left: 0.25rem;
}
.operation-switcher__chip-cta {
  font-size: 0.85rem;
  font-weight: 600;
  color: var(--color-primary, #2563eb);
  white-space: nowrap;
}
.operation-switcher__list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.operation-switcher__option {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  width: 100%;
  padding: 0.85rem 1rem;
  background: #fff;
  border: 1px solid rgba(15, 23, 42, 0.12);
  border-radius: var(--radius-md, 10px);
  font: inherit;
  text-align: left;
  cursor: pointer;
}
.operation-switcher__option:hover:not(:disabled) {
  border-color: var(--color-primary, #2563eb);
  background: rgba(37, 99, 235, 0.05);
}
.operation-switcher__option:disabled {
  cursor: default;
}
.operation-switcher__option.is-active {
  background: rgba(15, 23, 42, 0.04);
  border-color: rgba(15, 23, 42, 0.18);
}
.operation-switcher__option-name {
  flex: 1 1 auto;
  font-size: 1rem;
  font-weight: 600;
}
.operation-switcher__option-code {
  font-size: 0.78rem;
  color: var(--color-fg-muted);
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.operation-switcher__option-badge {
  font-size: 0.75rem;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  color: var(--color-fg-muted);
  border: 1px solid rgba(15, 23, 42, 0.15);
  padding: 0.1rem 0.4rem;
  border-radius: 999px;
}
/* Cutter-assistant /work теперь имеет два action-card одного уровня
   (см. `active-shift-panel.tsx` → `CutterAssistantWorkPanel`). На
   мобильном они идут вертикальной колонкой как и у швеи; gap 1rem
   достаточно, чтобы кнопки не сливались, но и не давали лишнего
   воздуха. Никаких grid/2-col — mobile-first. */
.cutter-assistant-work {
  gap: 1rem;
}
/* Shelf-placement session-режим: компактный заголовок с кодом ячейки
   крупно (helper не должен ошибиться, в какую ячейку он сейчас кладёт)
   и спокойная лента «Размещено» снизу. */
.shelf-placement__cell-label {
  display: block;
  font-size: 0.78rem;
  color: var(--color-fg-muted);
  text-transform: uppercase;
  letter-spacing: 0.06em;
  font-weight: 700;
  margin-bottom: 0.15rem;
}
.shelf-placement__cell-code {
  font-size: 1.6rem;
  letter-spacing: 0.02em;
  margin: 0 0 0.2rem;
}
/* Read-out режима после confirm-cell. Цель — за полсекунды показать
   помощнику, что больше QR ячейки сканировать не нужно: на пилоте
   видели, как оператор по привычке снова наводит камеру на стеллаж
   и упирается в общий PASSPORT_NOT_FOUND (см. `docs/flows.md §F3b`).
   Поэтому даём не подсказку курсивом, а явный inline-баннер. */
.shelf-placement__mode-banner {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  margin-top: 0.6rem;
  padding: 0.6rem 0.8rem;
  background: var(--color-ok-soft, #ecfdf5);
  color: var(--color-ok-fg, #065f46);
  border: 1px solid #a7f3d0;
  border-radius: var(--radius-md);
}
.shelf-placement__mode-title {
  font-size: 1rem;
  font-weight: 700;
  letter-spacing: 0.005em;
}
.shelf-placement__mode-sub {
  font-size: 0.85rem;
  opacity: 0.85;
}
.shelf-placement__recent {
  padding: 0.9rem 1rem;
}
.shelf-placement__recent-title {
  margin: 0 0 0.5rem;
  font-size: 0.9rem;
  text-transform: uppercase;
  letter-spacing: 0.05em;
  color: var(--color-fg-muted);
}
.shelf-placement__recent-list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.45rem;
}
.shelf-placement__recent-item {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 0.75rem;
  font-size: 0.95rem;
}
.shelf-placement__recent-meta {
  display: block;
  color: var(--color-fg-muted);
  font-size: 0.85rem;
  font-weight: 500;
  margin-top: 0.1rem;
}
.shelf-placement__recent-qty {
  font-weight: 700;
  white-space: nowrap;
}
.shelf-cell__qr {
  font-family: var(--font-mono, ui-monospace, SFMono-Regular, Menlo, monospace);
  font-size: 0.85rem;
  word-break: break-all;
}
.scan-card--simple {
  padding: 1.25rem 1.15rem;
  gap: 1rem;
}
.scan-card--simple .scan-card__title {
  font-size: 1.25rem;
}
.scan-card__primary-camera {
  min-height: 64px;
  font-size: 1.1rem;
  letter-spacing: 0.005em;
  display: inline-flex;
  align-items: center;
  justify-content: center;
}
/* Вторичная «Завершить операцию» (ТЗ §5): крупная, но визуально
   тише primary — та же высота и типографика, спокойная поверхность
   без акцентного заливания, чтобы не перетягивать внимание от
   главного действия «Взять крой». */
.scan-card__secondary-camera {
  min-height: 60px;
  font-size: 1.05rem;
  letter-spacing: 0.005em;
  font-weight: 600;
}
/* Вторичная action-кнопка под «Выпустить паспорт» помощника
   раскройщика (CutterAssistantWorkPanel). По форме совпадает с
   primary той же scan-card, но без акцентной заливки — открывает
   корректирующий flow «Выпущенные паспорта» (правка/удаление
   только что выпущенного паспорта). */
.scan-card__secondary-action {
  min-height: 56px;
  font-size: 1.05rem;
  font-weight: 600;
  letter-spacing: 0.005em;
  display: inline-flex;
  align-items: center;
  justify-content: center;
}
/* Список «Выпущенные паспорта» (`/work/passports`). Стиль mobile-
   first под /work — тёмная карточка на каждой строке, primary —
   крупный кликабельный номер паспорта, secondary-meta мелким
   текстом, кнопки справа. Заблокированные строки слегка приглушены,
   чтобы пользователь сразу видел: «здесь править нельзя». */
.my-passports-list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.65rem;
}
.my-passports-row {
  display: flex;
  flex-wrap: wrap;
  gap: 0.75rem;
  align-items: center;
  justify-content: space-between;
  padding: 0.85rem 1rem;
  background: var(--surface-2, #1f2330);
  border: 1px solid var(--border-1, #2a2f3d);
  border-radius: 14px;
}
.my-passports-row.is-blocked {
  opacity: 0.7;
}
.my-passports-row__main {
  display: flex;
  flex-direction: column;
  gap: 0.25rem;
  min-width: 0;
  flex: 1 1 240px;
}
.my-passports-row__number {
  font-size: 1.15rem;
  font-weight: 700;
  color: var(--text-1, #f1f3f7);
  text-decoration: underline;
  word-break: break-all;
}
.my-passports-row.is-blocked .my-passports-row__number {
  text-decoration: none;
}
.my-passports-row__meta {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  font-size: 0.95rem;
  color: var(--text-2, #c9ccd6);
}
.my-passports-row__sub {
  font-size: 0.8rem;
  color: var(--text-3, #8a90a2);
}
.my-passports-row__blocked-hint {
  font-size: 0.78rem;
  color: var(--color-warning-fg, #d6a300);
}
.my-passports-row__actions {
  display: flex;
  gap: 0.5rem;
  align-items: center;
  flex-wrap: wrap;
}
.scan-card__manual-toggle {
  appearance: none;
  background: transparent;
  border: none;
  padding: 0.4rem 0.5rem;
  font-size: 0.95rem;
  font-weight: 600;
  color: var(--color-accent);
  cursor: pointer;
  align-self: center;
  text-decoration: underline;
  text-underline-offset: 3px;
}
.scan-card__manual-toggle:hover { color: var(--color-accent-hover); }
.seamstress-work__switch {
  appearance: none;
  background: transparent;
  border: none;
  padding: 0.5rem 0.5rem;
  font-size: 0.9rem;
  font-weight: 500;
  color: var(--color-fg-muted);
  cursor: pointer;
  align-self: center;
  text-decoration: underline;
  text-underline-offset: 3px;
}
.seamstress-work__switch:hover { color: var(--color-fg); }
/* Три-точечное меню действий швеи (заменило большую красную кнопку
   «Завершить смену»). Кнопка-тригер прижата к правому верхнему углу
   страницы /work и не конкурирует с RoleHeaderCard визуально. */
.seamstress-actions {
  position: absolute;
  top: calc(env(safe-area-inset-top, 0px) + 0.4rem);
  right: 0.4rem;
  z-index: 6;
}
.seamstress-actions__trigger {
  appearance: none;
  background: rgba(255, 255, 255, 0.18);
  color: #fff;
  border: 1px solid rgba(255, 255, 255, 0.28);
  width: 40px;
  height: 40px;
  border-radius: 999px;
  font-size: 1.25rem;
  line-height: 1;
  cursor: pointer;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  transition: background 0.15s, border-color 0.15s;
}
.seamstress-actions__trigger:hover,
.seamstress-actions__trigger:focus-visible {
  background: rgba(255, 255, 255, 0.28);
  border-color: rgba(255, 255, 255, 0.45);
  outline: none;
}
.seamstress-actions__menu {
  position: absolute;
  top: calc(100% + 0.35rem);
  right: 0;
  min-width: 200px;
  background: #fff;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  box-shadow: var(--shadow-md);
  padding: 0.35rem;
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
}
.seamstress-actions__item {
  appearance: none;
  background: transparent;
  border: none;
  text-align: left;
  width: 100%;
  padding: 0.65rem 0.75rem;
  font-size: 0.95rem;
  font-weight: 500;
  color: var(--color-fg);
  border-radius: var(--radius-sm);
  cursor: pointer;
  min-height: 44px;
}
.seamstress-actions__item:hover,
.seamstress-actions__item:focus-visible {
  background: var(--color-bg-muted);
  outline: none;
}
.seamstress-actions__item--danger { color: var(--color-danger); }
.seamstress-actions__item--danger:hover,
.seamstress-actions__item--danger:focus-visible {
  background: rgba(220, 38, 38, 0.08);
}
.seamstress-actions__item:disabled {
  opacity: 0.55;
  cursor: not-allowed;
}
.seamstress-actions__error {
  padding: 0.4rem 0.75rem;
  font-size: 0.8rem;
  color: var(--color-danger);
}

/* =============================================================
   /work/cut-orders — упрощённый выбор заказа для CUTTER_ASSISTANT
   ============================================================= */
.cut-orders {
  display: flex;
  flex-direction: column;
  gap: 1.1rem;
  padding-top: calc(env(safe-area-inset-top, 0px));
}
.cut-orders__head {
  display: flex;
  align-items: center;
  gap: 0.75rem;
  min-height: 44px;
}
.cut-orders__back {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 44px;
  height: 44px;
  border-radius: 999px;
  background: var(--color-bg-muted);
  color: var(--color-fg);
  font-size: 1.25rem;
  text-decoration: none;
  border: 1px solid var(--color-border);
}
.cut-orders__back:hover,
.cut-orders__back:focus-visible {
  background: var(--color-bg-tint);
  outline: none;
}
.cut-orders__title {
  margin: 0;
  font-size: 1.25rem;
  font-weight: 700;
  letter-spacing: -0.01em;
  color: var(--color-fg);
}
.cut-orders__list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.65rem;
}
.cut-orders__card {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
  padding: 1rem 1.1rem;
  min-height: 72px;
  background: #fff;
  border: 1.5px solid var(--color-border-strong);
  border-radius: var(--radius-lg);
  text-decoration: none;
  color: var(--color-fg);
  box-shadow: var(--shadow-sm, 0 1px 2px rgba(15, 23, 42, 0.04));
  transition: border-color 0.15s, background 0.15s, transform 0.05s;
}
.cut-orders__card:hover,
.cut-orders__card:focus-visible {
  border-color: var(--color-accent);
  background: var(--color-bg-tint);
  outline: none;
}
.cut-orders__card:active {
  transform: scale(0.998);
}
.cut-orders__card-title {
  font-size: 1.05rem;
  font-weight: 700;
  line-height: 1.2;
}
.cut-orders__card-meta {
  display: inline-flex;
  gap: 0.4rem;
  align-items: baseline;
  font-size: 0.85rem;
  color: var(--color-fg-muted);
}
.cut-orders__card-number {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-weight: 600;
}
.cut-orders__empty {
  display: flex;
  flex-direction: column;
  gap: 0.65rem;
  padding: 1.5rem 1.25rem;
  background: #fff;
  border: 1px dashed var(--color-border-strong);
  border-radius: var(--radius-lg);
  text-align: center;
}
.cut-orders__empty-title {
  font-size: 1.05rem;
  font-weight: 700;
  color: var(--color-fg);
}
.cut-orders__empty-hint {
  margin: 0;
  font-size: 0.9rem;
  color: var(--color-fg-muted);
}

/* Старт смены: сводка по выбранному оборудованию + список операций */
.seamstress-start__manual {
  display: flex;
  flex-direction: column;
  gap: 0.65rem;
}
.seamstress-start__equipment {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  padding: 0.75rem 0.95rem;
  background: var(--color-bg-tint);
  border: 1px solid #c5d8ff;
  border-radius: var(--radius-md);
}
.seamstress-start__equipment-label {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 700;
  color: var(--color-accent-fg);
}
.seamstress-start__equipment-name {
  font-size: 1.1rem;
  font-weight: 700;
  color: var(--color-fg);
  line-height: 1.2;
}
.seamstress-start__equipment-meta {
  font-size: 0.8rem;
  color: var(--color-fg-muted);
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
}
.seamstress-start__ops {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.seamstress-start__op {
  display: grid;
  grid-template-columns: auto 1fr auto;
  align-items: center;
  gap: 0.65rem;
  padding: 0.85rem 1rem;
  border: 1.5px solid var(--color-border-strong);
  border-radius: var(--radius-md);
  background: #fff;
  cursor: pointer;
  min-height: 56px;
  transition: border-color 0.15s, background 0.15s, box-shadow 0.15s;
}
.seamstress-start__op:hover {
  border-color: var(--color-accent);
}
.seamstress-start__op input[type="radio"] {
  width: 22px;
  height: 22px;
  accent-color: var(--color-accent);
  margin: 0;
}
.seamstress-start__op-name {
  font-weight: 600;
  color: var(--color-fg);
  font-size: 1rem;
}
.seamstress-start__op-meta {
  font-size: 0.75rem;
  color: var(--color-fg-muted);
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
}
.seamstress-start__op.is-active {
  border-color: var(--color-accent);
  background: var(--color-bg-tint);
  box-shadow: 0 6px 16px -10px rgba(37, 99, 235, 0.45);
}

/* Success-state «Крой принят» */
.seamstress-success {
  background: #fff;
  border: 1.5px solid #bbf7d0;
  box-shadow: 0 10px 24px -16px rgba(22, 163, 74, 0.45);
}
.seamstress-success__head {
  display: flex;
  align-items: center;
  gap: 0.75rem;
}
.seamstress-success__check {
  width: 38px;
  height: 38px;
  border-radius: 50%;
  background: var(--color-ok);
  color: #fff;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  font-size: 1.3rem;
  font-weight: 700;
}
.seamstress-success__grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(120px, 1fr));
  gap: 0.5rem 1rem;
  margin: 0;
}
.seamstress-success__grid > div {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
}
.seamstress-success__grid dt {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  color: var(--color-fg-muted);
  font-weight: 700;
  margin: 0;
}
.seamstress-success__grid dd {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 700;
  color: var(--color-fg);
}

/* =========================================================================
   Блок «Текущий крой» / «Сейчас в работе» (CurrentWorkCard)
   -------------------------------------------------------------------------
   Mobile-first. Используется на /work для SEAMSTRESS, чтобы у швеи был
   устойчивый рабочий контекст после приёма кроя (см. docs/screens.md §3.3
   и docs/flows.md §F3a). Без таблиц, крупные значения, приглушённые
   labels — единый язык с .seamstress-success и .passport-confirm.
   ========================================================================= */
.current-work {
  display: flex;
  flex-direction: column;
  gap: 0.65rem;
  padding: 0.95rem 1rem 1.1rem;
  background: var(--color-bg-card);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  box-shadow: 0 8px 22px -18px rgba(15, 23, 42, 0.35);
}
.current-work--empty {
  background: var(--color-bg-muted);
  border-style: dashed;
  border-color: var(--color-border-strong);
  box-shadow: none;
}
.current-work__head {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.75rem;
}
/* Кликабельная шапка-тогл (`<button>`-вариант) — сворачивает список
   паспортов под нею. Сбрасываем дефолтные стили кнопки, чтобы внешне
   шапка осталась идентичной неинтерактивной (см. current-work-card.tsx). */
button.current-work__head--toggle {
  width: 100%;
  background: transparent;
  border: 0;
  padding: 0;
  margin: 0;
  font: inherit;
  color: inherit;
  text-align: left;
  cursor: pointer;
}
button.current-work__head--toggle:hover .current-work__title,
button.current-work__head--toggle:focus-visible .current-work__title {
  color: var(--color-fg);
}
.current-work__caret {
  flex-shrink: 0;
  font-size: 0.9rem;
  line-height: 1;
  color: var(--color-fg-muted);
  margin-left: 0.25rem;
}
.current-work__title {
  margin: 0;
  font-size: 0.78rem;
  font-weight: 700;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  color: var(--color-fg-muted);
}
.current-work__summary {
  display: inline-flex;
  align-items: baseline;
  gap: 0.4rem;
  flex-shrink: 0;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  line-height: 1;
}
.current-work__summary-units {
  font-size: 1rem;
  font-weight: 700;
  color: var(--color-fg);
}
.current-work__summary-sep {
  color: var(--color-fg-muted);
  font-weight: 500;
}
.current-work__summary-passports {
  font-size: 0.95rem;
  font-weight: 500;
  color: var(--color-fg-muted);
}
.current-work__empty-text {
  margin: 0;
  color: var(--color-fg-muted);
  font-size: 0.92rem;
  line-height: 1.4;
}
.current-work__list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.7rem;
}

.active-passport {
  display: flex;
  flex-direction: column;
  gap: 0.55rem;
  padding: 0.85rem 0.95rem 0.95rem;
  border: 1px solid var(--color-border);
  border-left: 4px solid var(--color-accent);
  border-radius: var(--radius-md);
  background: #fff;
}
.active-passport__head {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 0.5rem;
}
/* Чекбокс мультивыбора для пакетного завершения. Сидит слева от
   номера паспорта; номер получает margin-right:auto, чтобы блок
   «Из маршрута / принят» уезжал к правому краю как раньше. */
.active-passport__select {
  display: inline-flex;
  align-items: center;
  flex-shrink: 0;
  align-self: center;
  cursor: pointer;
}
.active-passport__select input {
  width: 1.3rem;
  height: 1.3rem;
  accent-color: var(--color-accent, #2f6fed);
  cursor: pointer;
}
.active-passport__head .active-passport__number {
  margin-right: auto;
}
/* Подсветка выбранной карточки и карточки, которая не закрылась при
   пакетном завершении (партиальный успех). */
.active-passport--selected {
  border-left-color: var(--color-accent, #2f6fed);
  background: var(--color-bg-tint, #f5f8ff);
}
.active-passport--failed {
  border-left-color: var(--color-danger, #d33);
}
.active-passport__failed {
  margin: 0;
  font-size: 0.85rem;
  color: var(--color-danger, #c0392b);
}
/* Тулбар «Выбрать все» над списком и кнопка «Завершить выбранные». */
.current-work__select-bar {
  display: flex;
  align-items: center;
  margin-bottom: 0.6rem;
}
.current-work__select-all {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  font-size: 0.9rem;
  font-weight: 600;
  cursor: pointer;
  -webkit-user-select: none;
          user-select: none;
}
.current-work__select-all input {
  width: 1.2rem;
  height: 1.2rem;
  accent-color: var(--color-accent, #2f6fed);
  cursor: pointer;
}
.current-work__actions {
  margin-top: 0.85rem;
}
/* Модалка-сводка пакетного завершения — действия в ряд. */
.modal--batch-complete .modal__actions {
  display: flex;
  gap: 0.6rem;
  justify-content: flex-end;
}
/* Группа справа от номера паспорта: баджик «Из маршрута» (для
   route-WIP, см. `docs/domain.md §«Маршруты производства»») и время
   приёма. Стояли раньше отдельным span-ом; обернули в flex-контейнер,
   чтобы оба не разъезжались по строкам, когда номер паспорта длинный. */
.active-passport__head-meta {
  display: inline-flex;
  align-items: baseline;
  gap: 0.4rem;
  flex-shrink: 0;
}
/* Спокойный inline-бейдж «Из маршрута»: подсказывает оператору, что
   паспорт идёт по маршрутному потоку (`currentRouteStepIndex !== null`)
   и приём не требует размещения в ячейке. Используется одновременно в
   карточке активного кроя и в `PassportConfirmModal` — стилистика
   общая. Цвет совпадает с `.active-passport__route` блоком, чтобы
   визуально читался как «то же самое, маршрут». */
.active-passport__route-badge,
.passport-confirm__route-badge {
  display: inline-flex;
  align-items: center;
  flex-shrink: 0;
  padding: 0.18rem 0.5rem;
  font-size: 0.7rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--color-accent-fg, #1d3da3);
  background: var(--color-bg-tint, #eef2ff);
  border: 1px solid var(--color-border-subtle, #c5d8ff);
  border-radius: 999px;
  line-height: 1.1;
  white-space: nowrap;
}
.passport-confirm__route-badge {
  align-self: flex-start;
  margin-top: 0.25rem;
}
/* Спокойный subtext в `PassportConfirmModal` для случая
   route-WIP без ячейки: вместо «ячейка не указана» показываем
   «маршрутный поток — ячейка не требуется», без alert-семантики. */
.passport-confirm__route-subtext {
  margin: -0.25rem 0 0;
  font-size: 0.85rem;
  line-height: 1.3;
  color: var(--color-fg-muted);
}
.active-passport__number {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 1.2rem;
  font-weight: 700;
  color: var(--color-fg);
  text-decoration: none;
  word-break: break-all;
}
.active-passport__number:hover {
  color: var(--color-accent);
}
.active-passport__accepted {
  flex-shrink: 0;
  font-size: 0.72rem;
  font-weight: 600;
  color: var(--color-fg-muted);
  text-transform: lowercase;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.active-passport__product {
  font-size: 1rem;
  font-weight: 600;
  color: var(--color-fg);
  line-height: 1.3;
}
.active-passport__color {
  color: var(--color-fg-muted);
  font-weight: 500;
}
.active-passport__grid {
  display: grid;
  grid-template-columns: repeat(2, 1fr);
  gap: 0.55rem 1rem;
  margin: 0;
}
.active-passport__grid > div {
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  min-width: 0;
}
.active-passport__grid dt {
  margin: 0;
  font-size: 0.68rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 700;
  color: var(--color-fg-muted);
}
.active-passport__grid dd {
  margin: 0;
  font-size: 0.98rem;
  font-weight: 600;
  color: var(--color-fg);
  line-height: 1.25;
  word-break: break-word;
}
.active-passport__size {
  font-size: 1.2rem !important;
  font-weight: 700 !important;
}
.active-passport__qty { font-feature-settings: "tnum"; font-variant-numeric: tabular-nums; font-size: 1.05rem; }
.active-passport__qty-meta { color: var(--color-fg-muted); font-weight: 500; font-size: 0.85rem; }
.active-passport__op { font-size: 0.95rem; }
.active-passport__op-code {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 0.78rem;
  color: var(--color-fg-muted);
  font-weight: 500;
}

@media (max-width: 380px) {
  .active-passport__grid { grid-template-columns: 1fr; }
}

/* Soft-route hint (см. docs/domain.md §«Маршруты производства»):
   текущий и следующий шаг маршрута, плюс warning при mismatch с
   операцией смены. На MVP — без блокировки. */
.active-passport__route {
  margin-top: 0.6rem;
  display: flex;
  flex-direction: column;
  gap: 0.25rem;
  padding: 0.55rem 0.7rem;
  background: var(--color-bg-tint, #f6f8ff);
  border: 1px solid var(--color-border-subtle, #dbe3f3);
  border-radius: var(--radius-md, 8px);
}
.active-passport__route-row {
  display: flex;
  gap: 0.5rem;
  align-items: baseline;
  font-size: 0.92rem;
  line-height: 1.25;
}
.active-passport__route-label {
  flex: 0 0 3.4rem;
  font-size: 0.75rem;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--color-fg-muted);
  font-weight: 600;
}
.active-passport__route-value {
  color: var(--color-fg);
  font-weight: 500;
  word-break: break-word;
}
.active-passport__route-code {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 0.78rem;
  color: var(--color-fg-muted);
  font-weight: 500;
}
.active-passport__route-warn {
  margin: 0.2rem 0 0;
  padding: 0.4rem 0.55rem;
  background: #fff7e0;
  border: 1px solid #f0d27a;
  border-radius: 6px;
  font-size: 0.88rem;
  color: #6b4d00;
  line-height: 1.3;
}
.active-passport__route-warn strong { font-weight: 700; }
.active-passport__route-warn em { font-style: normal; font-weight: 600; }

/* Модалка проверки паспорта (PassportConfirmModal) */
.passport-confirm__card { gap: 0.85rem; padding: 1.1rem 1.1rem 1.2rem; }
.passport-confirm__number {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  padding: 0.85rem 1rem;
  background: var(--color-bg-tint);
  border: 1px solid #c5d8ff;
  border-radius: var(--radius-md);
}
.passport-confirm__number-label {
  font-size: 0.72rem;
  letter-spacing: 0.08em;
  text-transform: uppercase;
  font-weight: 700;
  color: var(--color-accent-fg);
}
.passport-confirm__number-value {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 1.45rem;
  font-weight: 700;
  color: var(--color-fg);
  word-break: break-all;
  line-height: 1.15;
}
.passport-confirm__grid {
  display: grid;
  grid-template-columns: repeat(2, 1fr);
  gap: 0.7rem 1rem;
  margin: 0;
}
.passport-confirm__grid > div { display: flex; flex-direction: column; gap: 0.1rem; }
.passport-confirm__grid dt {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 700;
  color: var(--color-fg-muted);
  margin: 0;
}
.passport-confirm__grid dd {
  margin: 0;
  font-size: 1rem;
  font-weight: 600;
  color: var(--color-fg);
}
.passport-confirm__size { font-size: 1.4rem !important; font-weight: 700 !important; }
.passport-confirm__qty { font-feature-settings: "tnum"; font-variant-numeric: tabular-nums; }
.passport-confirm__qty-meta { color: var(--color-fg-muted); font-weight: 500; font-size: 0.85rem; }
.passport-confirm__shift {
  display: grid;
  grid-template-columns: repeat(2, 1fr);
  gap: 0.5rem 1rem;
  padding: 0.7rem 0.85rem;
  background: var(--color-bg-muted);
  border-radius: var(--radius-md);
}
.passport-confirm__shift > div { display: flex; flex-direction: column; gap: 0.1rem; }
.passport-confirm__shift-label {
  font-size: 0.65rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 700;
  color: var(--color-fg-muted);
}
.passport-confirm__shift-value {
  font-size: 0.95rem;
  font-weight: 600;
  color: var(--color-fg);
  line-height: 1.2;
}
.passport-confirm__shift-meta {
  font-size: 0.72rem;
  color: var(--color-fg-muted);
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
}
.passport-confirm__actions {
  flex: 0 0 auto;
  display: grid;
  grid-template-columns: 1fr 1.4fr;
  gap: 0.5rem;
}
.passport-confirm__actions .btn { min-height: 52px; }

@media (max-width: 380px) {
  .passport-confirm__grid { grid-template-columns: 1fr; }
  .passport-confirm__shift { grid-template-columns: 1fr; }
  .passport-confirm__actions { grid-template-columns: 1fr; }
}

@media (min-width: 600px) {
  .qr-modal { padding: 1.5rem; }
}

/* ----------------------------------------------------------------- */
/* QC role-terminal: рабочая карточка паспорта на /qc                */
/* (см. apps/web/app/qc/qc-work-card.tsx, scan-driven flow F5).      */
/* Стилизация повторяет .active-passport, чтобы UX ОТК и швеи        */
/* визуально совпадал.                                               */
/* ----------------------------------------------------------------- */
.qc-card {
  display: flex;
  flex-direction: column;
  gap: 0.75rem;
  padding: 0.95rem 1rem 1.05rem;
  border: 1px solid var(--color-border);
  border-left: 4px solid var(--color-accent);
  border-radius: var(--radius-md);
  background: #fff;
}
.qc-card__header {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 0.5rem;
}
.qc-card__label {
  font-size: 0.68rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 700;
  color: var(--color-fg-muted);
}
.qc-card__number {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 1.2rem;
  font-weight: 700;
  color: var(--color-fg);
  word-break: break-all;
}
.qc-card__grid {
  display: grid;
  grid-template-columns: repeat(2, 1fr);
  gap: 0.55rem 1rem;
  margin: 0;
}
.qc-card__grid > div {
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  min-width: 0;
}
.qc-card__grid dt {
  margin: 0;
  font-size: 0.68rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 700;
  color: var(--color-fg-muted);
}
.qc-card__grid dd {
  margin: 0;
  font-size: 0.98rem;
  font-weight: 600;
  color: var(--color-fg);
  line-height: 1.25;
  word-break: break-word;
}
.qc-card__size { font-size: 1.2rem !important; font-weight: 700 !important; }
.qc-card__qty {
  display: grid;
  grid-template-columns: repeat(3, 1fr);
  gap: 0.4rem;
  padding: 0.55rem 0.6rem;
  background: var(--color-bg-tint);
  border-radius: var(--radius-md);
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.qc-card__qty > div {
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 0.1rem;
}
.qc-card__qty-label {
  font-size: 0.68rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 700;
  color: var(--color-fg-muted);
}
.qc-card__qty-value { font-size: 1.25rem; font-weight: 700; color: var(--color-fg); }
.qc-card__qty-value--danger { color: #b00020; }
.qc-card__qty-value--success { color: #137333; }
.qc-card__defect-form {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  padding-top: 0.4rem;
  border-top: 1px dashed var(--color-border);
}
.qc-card__defect-empty {
  padding: 0.6rem 0.7rem;
  font-size: 0.9rem;
  color: var(--color-fg-muted);
}
.qc-card__section-title {
  margin: 0;
  font-size: 0.95rem;
  font-weight: 700;
  color: var(--color-fg);
}
.qc-card__history { font-size: 0.9rem; }
.qc-card__history summary {
  cursor: pointer;
  font-weight: 600;
  color: var(--color-fg-muted);
}
.qc-card__history ul {
  list-style: none;
  margin: 0.5rem 0 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
}
.qc-card__history-meta { color: var(--color-fg-muted); font-size: 0.85rem; }
.qc-card__history-comment {
  margin-top: 0.15rem;
  color: var(--color-fg-muted);
  font-size: 0.85rem;
}
/* Закреплённая снизу панель действий ОТК (см.
   apps/web/app/qc/qc-work-card.tsx). Прилипает к нижней кромке
   вьюпорта, пока тело карточки скроллится; набор кнопок меняется
   после «Проверка выполнена». Отрицательные поля растягивают панель
   на всю ширину карточки поверх её паддингов. */
.qc-card__sticky-actions {
  position: sticky;
  bottom: 0;
  z-index: 5;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  margin: 0.4rem -1rem -1.05rem;
  padding: 0.7rem 1rem calc(0.7rem + env(safe-area-inset-bottom, 0px));
  background: #fff;
  border-top: 1px solid var(--color-border);
}
/* «Обновить карточку» — мелкая ссылка, прижата вправо, вне панели. */
.qc-card__refresh {
  display: flex;
  justify-content: flex-end;
  margin-top: -0.35rem;
}

@media (max-width: 380px) {
  .qc-card__grid { grid-template-columns: 1fr; }
}

/* QC role-terminal: компактная строка «Проверено ОТК» (см.
   apps/web/app/qc/qc-completed-row.tsx). Появляется после нажатия
   «Проверка выполнена» вместо большой рабочей карточки и исчезает,
   когда backend отдаёт `removedFromQc=true` (паспорт ушёл дальше). */
.qc-done-row {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  justify-content: space-between;
  gap: 0.55rem 0.75rem;
  padding: 0.65rem 0.85rem;
  border: 1px solid var(--color-border);
  border-left: 4px solid #137333;
  border-radius: var(--radius-md);
  background: #fff;
}
.qc-done-row__main {
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  min-width: 0;
}
.qc-done-row__number {
  font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, monospace;
  font-size: 0.95rem;
  font-weight: 700;
  color: var(--color-fg);
  word-break: break-all;
}
.qc-done-row__meta {
  display: flex;
  align-items: baseline;
  gap: 0.4rem;
  color: var(--color-fg-muted);
  font-size: 0.85rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.qc-done-row__size { font-weight: 700; color: var(--color-fg); }
.qc-done-row__qty { font-weight: 600; color: var(--color-fg); }
.qc-done-row__sep { color: var(--color-border); }
.qc-done-row__badge {
  display: inline-flex;
  align-items: center;
  padding: 0.2rem 0.55rem;
  border-radius: 999px;
  background: #e6f4ea;
  color: #0f5132;
  font-size: 0.78rem;
  font-weight: 700;
  letter-spacing: 0.02em;
  white-space: nowrap;
}

/* QC role-terminal: компактная кнопка «Выйти» в правом верхнем углу
   (глобальная шапка на /qc спрятана у роли QC, нужен запасной выход
   из аккаунта — см. apps/web/app/qc/qc-terminal.tsx). */
.qc-logout {
  align-self: flex-end;
  margin: 0;
}
.qc-logout__btn {
  appearance: none;
  background: transparent;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-sm);
  padding: 0.35rem 0.75rem;
  font-size: 0.85rem;
  font-weight: 600;
  color: var(--color-fg-muted);
  cursor: pointer;
}
.qc-logout__btn:hover { color: var(--color-fg); border-color: var(--color-fg-muted); }

/* =============================================================
   Bulk print «Печать всех ячеек» (см. apps/web/app/admin/warehouses/[id]/bulk-print-panel.tsx,
   docs/screens.md §10b). Модалка использует базовые `.qr-modal*`
   стили, плюс свои блоки настроек/превью/итогов.
   ============================================================= */
.section-header__actions {
  display: inline-flex;
  align-items: center;
  gap: 0.65rem;
  flex-wrap: wrap;
}
/*
 * Bulk-print карточка: bounded по dvh, без самостоятельного scroll
 * (он живёт на `__body`). Card сама — flex-column-контейнер для
 * header / form / footer; padding обнулён, потому что внутренние
 * секции дают свой horizontal padding (header — от `.qr-modal__card`
 * 1rem, body/footer — 1.1rem).
 */
.bulk-print-modal__card {
  width: min(100%, 720px);
  max-width: none;
  /* `vh`-fallback для Safari < 15.4 (см. `.auth-screen`). */
  max-height: calc(100vh - 2rem);
  max-height: calc(100dvh - 2rem);
  padding: 0;
  gap: 0;
  display: flex;
  flex-direction: column;
  overflow: hidden;
}
/* Header не растёт и не ужимается — фикс-высота под title + close. */
.bulk-print-modal__card > .qr-modal__header {
  flex: 0 0 auto;
  padding: 1rem 1.1rem 0;
}
/*
 * Form — flex-column контейнер для body. Сама форма не скроллится,
 * скролл внутри `__body`. Это даёт single-scroll-context UX: одна
 * полоса прокрутки внутри карточки, footer всегда видим.
 */
.bulk-print-modal__form {
  flex: 1 1 auto;
  min-height: 0;
  display: flex;
  flex-direction: column;
  overflow: hidden;
}
/* Body — единственный scrollable child карточки. */
.bulk-print-modal__body {
  flex: 1 1 auto;
  min-height: 0;
  overflow-y: auto;
  display: flex;
  flex-direction: column;
  gap: 1rem;
  padding: 1rem 1.1rem 0.5rem;
}
/* Footer — снаружи формы (submit-кнопка через `form={id}`). */
.bulk-print-modal__footer {
  flex: 0 0 auto;
  display: flex;
  align-items: center;
  justify-content: flex-end;
  gap: var(--admin-space-sm, 0.6rem);
  flex-wrap: wrap;
  padding: 0.7rem 1.1rem 1rem;
  border-top: 1px solid var(--color-border);
  background: #fff;
}
.bulk-print-modal__summary {
  display: grid;
  grid-template-columns: repeat(3, 1fr);
  gap: 0.5rem;
  padding: 0.75rem 0.85rem;
  background: var(--color-bg-muted, #f7f8fa);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
}
.bulk-print-modal__summary > div {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
}
.bulk-print-modal__summary-label {
  font-size: 0.72rem;
  text-transform: uppercase;
  letter-spacing: 0.06em;
  color: var(--color-fg-muted);
  font-weight: 600;
}
.bulk-print-modal__summary-value {
  font-size: 1.25rem;
  font-weight: 800;
  color: var(--color-fg-strong, var(--color-fg));
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.bulk-print-modal__preview {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.bulk-print-modal__preview-header {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 0.5rem;
  flex-wrap: wrap;
  font-weight: 600;
  color: var(--color-fg);
}
.bulk-print-modal__preview-grid {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(160px, 1fr));
  gap: 0.55rem;
  max-height: 32vh;
  overflow-y: auto;
  padding: 0.4rem;
  background: #f1f3f7;
  border-radius: var(--radius-md);
  border: 1px solid var(--color-border);
}
/* Плитка-этикетка повторяет пропорции 58:38 ≈ 1.53 — глаз менеджера
   сразу узнаёт реальную термоэтикетку. Чёрно-белая, минимум декора. */
.bulk-print-modal__label {
  display: flex;
  flex-direction: row;
  align-items: stretch;
  background: #fff;
  color: #000;
  border: 1px solid #d4d8df;
  border-radius: 6px;
  aspect-ratio: 58 / 38;
  overflow: hidden;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.06);
}
.bulk-print-modal__label-qr {
  flex: 1 1 50%;
  display: flex;
  align-items: center;
  justify-content: center;
  padding: 4px;
  background: #fff;
}
.bulk-print-modal__label-qr img {
  width: 100%;
  height: 100%;
  max-width: 100%;
  max-height: 100%;
  object-fit: contain;
  image-rendering: pixelated;
}
.bulk-print-modal__label-code {
  flex: 1 1 50%;
  display: flex;
  align-items: center;
  justify-content: center;
  font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, monospace;
  font-weight: 900;
  font-size: 1.5rem;
  letter-spacing: -0.02em;
  word-break: break-word;
  text-align: center;
  padding: 0 4px;
  line-height: 1;
}
@media (max-width: 640px) {
  .bulk-print-modal__summary {
    grid-template-columns: 1fr;
  }
}

/* ============================================================================
   Shopfloor display screen — большой монитор в зале (Production Board, Шаг 10b).
   Цели:
   - светлая тема, аккуратный «дашборд»-look (бывшая тёмная тема ушла,
     см. историю: `apps/web/app/shopfloor/display/display-board.tsx`);
   - layout: header → KPI-row → grid (production matrix 2/3 + equipment 1/3);
   - производственная матрица — главный блок: «цвет × размер × stage»;
   - equipment — компактные плитки с иконкой типа и номером (≈ 30 шт.
     помещаются без скролла на 1080p);
   - никаких hover/focus-состояний и интерактива — экран read-only.
   ============================================================================ */
.display-screen {
  /* Light dashboard palette — намеренно изолировано от глобальной темы
     приложения (только под `.display-screen`), чтобы остальной UI не
     переключался в светлый режим. */
  --display-bg: #f1f5f9;
  --display-bg-soft: #f8fafc;
  --display-bg-card: #ffffff;
  --display-fg: #0f172a;
  --display-fg-muted: #475569;
  --display-fg-subtle: #94a3b8;
  --display-accent: #2563eb;
  --display-accent-soft: #dbeafe;
  --display-ok: #16a34a;
  --display-ok-soft: #dcfce7;
  --display-warn: #f59e0b;
  --display-warn-soft: #fef3c7;
  --display-crit: #dc2626;
  --display-crit-soft: #fee2e2;
  --display-border: #e2e8f0;
  --display-border-strong: #cbd5e1;
  /* Подсветка узкого места в split-блоках «Поток производства».
     Цвет намеренно coral (`#fb7185`), а не алертный `--display-crit`
     (#dc2626): «бутылочное горлышко» — это сигнал «обратите внимание»,
     а не «авария». Coral хорошо контрастирует с нейтральным фоном
     матрицы (`#ffffff` / total-row `#f8fafc`) и не сливается с
     красным flash'ем изменения значения (`rgba(255,0,0,…)`). */
  --display-bottleneck-color: #fb7185;
  --display-shadow: 0 1px 2px rgba(15, 23, 42, 0.04),
    0 4px 12px rgba(15, 23, 42, 0.04);
  position: fixed;
  inset: 0;
  /* Высота: сначала `100vh` как fallback, потом `100dvh` поверх него.
     Зачем именно две строки в таком порядке:
       - старые embedded WebView (TV / Tizen / WebKit < 108) НЕ
         понимают `dvh`-юнит, и без fallback'а правило целиком
         игнорировалось бы — `position: fixed; inset: 0` всё равно
         даёт высоту через bottom:0, но на некоторых TV-сборках
         браузер при этом всё-таки берёт `height: auto` и схлопывает
         board-row до content-height (видны только заголовки);
       - современные браузеры просто перезаписывают первую строку
         второй — поведение `100dvh` сохраняется как раньше
         (стабильно при появлении/уходе address-bar/UI overlay).
     Важно: cascade'е порядок имеет значение — `100dvh` ДОЛЖЕН идти
     после `100vh`, иначе fallback перезатрёт прогрессивное правило. */
  width: 100vw;
  height: 100vh;
  height: 100dvh;
  display: grid;
  /* header (auto) → KPI (auto) → board (1fr, занимает остаток).
     `minmax(0, 1fr)` гарантирует, что board строка может «сжиматься»
     до своего min-content, а не раздувать grid выше viewport — это
     ключ к надёжному внутреннему overflow матрицы. */
  grid-template-rows: auto auto minmax(0, 1fr);
  gap: 1rem;
  padding: 1.25rem 1.5rem 1.5rem;
  background: var(--display-bg);
  color: var(--display-fg);
  font-family: 'Inter', system-ui, sans-serif;
  /* Сама обёртка ничего не скроллит — скроллятся только внутренние
     зоны (`.display-matrix__scroll` и `.display-equipment-grid`). */
  overflow: hidden;
  z-index: 10;
}
.display-screen--light {
  /* Селектор-флаг: вариант светлой темы по умолчанию (другие варианты
     не предусмотрены, но изолируем класс на случай будущего toggle). */
  color-scheme: light;
}
.display-screen__header {
  display: grid;
  grid-template-columns: 1fr auto 1fr;
  align-items: center;
  gap: 1rem;
  padding: 0.25rem 0;
}
.display-screen__brand {
  font-size: clamp(1.2rem, 1.6vw, 1.6rem);
  font-weight: 700;
  letter-spacing: 0.18em;
  display: inline-flex;
  align-items: center;
  gap: 0.6rem;
  color: var(--display-fg);
}
.display-screen__brand-mark {
  color: var(--display-accent);
  font-size: 1.5em;
  line-height: 1;
}
.display-screen__clock {
  font-size: clamp(2rem, 3.4vw, 3.4rem);
  font-weight: 700;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  text-align: center;
  letter-spacing: 0.05em;
  color: var(--display-fg);
}
.display-screen__meta {
  display: inline-flex;
  align-items: center;
  justify-content: flex-end;
  gap: 1rem;
  font-size: clamp(0.85rem, 1vw, 1rem);
  color: var(--display-fg-muted);
}
.display-status {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  text-transform: uppercase;
  letter-spacing: 0.12em;
  font-weight: 600;
}
.display-status__dot {
  width: 0.65rem;
  height: 0.65rem;
  border-radius: 50%;
  background: currentColor;
}
.display-status--online {
  color: var(--display-ok);
}
/* «Обновление замедлено» — есть soft-ошибки, но snapshot жив и
   polling переведён на degraded-cadence (см. POLL_INTERVAL_DEGRADED_MS
   в `display-board.tsx`). Жёлтый = «осторожно, но не критично». */
.display-status--degraded {
  color: var(--display-warn);
}
.display-status--offline {
  color: var(--display-crit);
}
/* Auth-проблема (HTTP 401/403). Сигнал визуально близок к
   degraded — корень проблемы в сессии, а не в сети, но snapshot
   на экране остаётся валидным. */
.display-status--auth {
  color: var(--display-warn);
}
.display-screen__updated {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
/* Кнопка «Выйти» на мониторе. AppHeader (и его logout) на этом экране
   скрыт, поэтому выход живёт в шапке самой витрины. Спокойный outline-
   стиль, чтобы не конкурировать со статус-чипом и KPI. */
.display-screen__logout-form {
  display: inline-flex;
  margin: 0;
}
.display-screen__logout {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.3rem 0.7rem;
  border: 1px solid var(--display-fg-muted);
  border-radius: 0.5rem;
  background: transparent;
  color: var(--display-fg-muted);
  font: inherit;
  font-size: clamp(0.8rem, 0.95vw, 0.95rem);
  line-height: 1;
  cursor: pointer;
  transition:
    color 0.15s ease,
    border-color 0.15s ease,
    background 0.15s ease;
}
.display-screen__logout:hover {
  color: var(--display-crit);
  border-color: var(--display-crit);
  background: rgba(0, 0, 0, 0.03);
}

/* KPI row ------------------------------------------------------------------ */
.display-kpi {
  display: grid;
  grid-template-columns: repeat(8, minmax(0, 1fr));
  gap: 0.75rem;
}
@media (max-width: 1399px) {
  .display-kpi {
    grid-template-columns: repeat(4, minmax(0, 1fr));
  }
}
.display-kpi__card {
  background: var(--display-bg-card);
  border: 1px solid var(--display-border);
  border-radius: 0.85rem;
  padding: 0.85rem 1rem;
  display: flex;
  flex-direction: column;
  gap: 0.25rem;
  box-shadow: var(--display-shadow);
}
.display-kpi__value {
  font-size: clamp(1.6rem, 2.4vw, 2.4rem);
  font-weight: 800;
  line-height: 1.05;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--display-fg);
}
.display-kpi__label {
  font-size: 0.78rem;
  text-transform: uppercase;
  letter-spacing: 0.12em;
  color: var(--display-fg-muted);
  font-weight: 600;
}
.display-kpi__card--ok .display-kpi__value {
  color: var(--display-ok);
}
.display-kpi__card--ok {
  background: linear-gradient(180deg, var(--display-ok-soft), #ffffff 70%);
  border-color: #bbf7d0;
}
.display-kpi__card--accent .display-kpi__value {
  color: var(--display-accent);
}
.display-kpi__card--accent {
  background: linear-gradient(180deg, var(--display-accent-soft), #ffffff 70%);
  border-color: #bfdbfe;
}
.display-kpi__card--warn .display-kpi__value {
  color: var(--display-warn);
}
.display-kpi__card--crit .display-kpi__value {
  color: var(--display-crit);
}
.display-kpi__card--crit {
  background: linear-gradient(180deg, var(--display-crit-soft), #ffffff 70%);
  border-color: #fecaca;
}

/* Production board: матрица слева (растягивается), оборудование справа.
   Оба блока — независимые grid items, чтобы матрица могла внутри себя
   скроллиться без ломания equipment-плиток. */
.display-board {
  display: grid;
  grid-template-columns: minmax(0, 2fr) minmax(320px, 1fr);
  gap: 1rem;
  /* Цепочка `min-height: 0` (display-screen → display-board →
     production/equipment → display-block → scroll-area) ОБЯЗАТЕЛЬНА:
     иначе flex/grid items получают неявный `min-height: auto`,
     равный intrinsic height содержимого (длинная таблица), и тогда
     внешние `overflow: hidden` срезают матрицу так, что на экране
     остаются только заголовки. */
  min-height: 0;
  /* `min-width: 0` — симметричный fix для горизонтального overflow
     длинных рядов матрицы; без него grid item раздувается по
     content-width и ломает правую колонку equipment. */
  min-width: 0;
}
.display-board__production,
.display-board__equipment {
  display: flex;
  flex-direction: column;
  min-height: 0;
  min-width: 0;
}
.display-board__production > .display-block,
.display-board__equipment > .display-block {
  flex: 1 1 auto;
  min-height: 0;
  min-width: 0;
}
/* Sewing-операция в шапке матрицы — двухстрочная конструкция:
   row-1 — один <th colSpan=2> с именем операции (центрировано над
            своими подколонками);
   row-2 — два <th> с маркерами ▶ / ✔, стоящими РОВНО под именем.
   Имя и иконки разнесены по разным <tr>, чтобы оператор на TV сразу
   видел соответствие «операция → её две колонки», а не путал inline-
   формат «Оверлок 1 ▶ ✔» с цифрами строк ниже. */
.display-matrix__th--sewing-op {
  white-space: nowrap;
  text-align: center;
  /* Имя операции занимает обе подколонки; рамку рисуем только
     снизу — между ним и второй строкой шапки. */
  border-bottom: 1px solid var(--display-border-strong);
  padding-bottom: 0.4rem;
}
.display-matrix__th--sewing-op .display-matrix__th-op {
  display: inline-block;
  letter-spacing: 0.06em;
}
/* Вторая строка шапки (подколонки ▶/✔). Sticky должен «прилипать»
   к низу первой строки, иначе под ней просвечивают строки данных
   при скролле. `top` берётся с запасом по высоте row-1, рассчитан
   эмпирически по реальному `font-size 0.85rem + padding 0.65rem`
   первой строки. На TV-слое (≥1600px) подменяется ниже. */
.display-matrix__th--sub {
  top: 2.4rem;
  font-size: 1rem;
  font-weight: 800;
  letter-spacing: 0;
  padding: 0.35rem 0.4rem;
}
.display-matrix__th--sub .display-matrix__th-dir {
  display: inline-block;
  font-size: 1.1rem;
  font-weight: 800;
  line-height: 1;
}
/* Компактная ширина под одиночный символ ▶/✔ — обе подколонки
   одинаковые, чтобы данные строк ниже выравнивались зеркально. */
.display-matrix__th--sew-in,
.display-matrix__th--sew-done {
  min-width: 2.6rem;
  width: 2.6rem;
}
/* Внутренний разделитель между ▶ и ✔ убираем: визуально они
   принадлежат одной операции, граница рисуется только под
   именем операции и снаружи всей пары. */
.display-matrix__th--sew-done {
  border-left: 0;
}
.display-matrix__cell--sew-in {
  font-weight: 700;
}
.display-matrix__cell--sew-done {
  font-weight: 700;
  color: var(--display-ok);
  /* Защитный no-op на случай, если кто-нибудь позже навесит divider
     ещё и на ✔: визуально пара ▶/✔ принадлежит одной операции, и
     второй вертикальной линии между ними быть не должно. Активный
     `display-matrix__op-divider` всегда остаётся ТОЛЬКО на ▶ (см.
     `SewingOpSubHeader` и body-/totals-ячейки). */
  border-left: none;
}
/* Тонкая «спокойная» вертикальная линия, отделяющая одну sewing-
   операцию от другой. Навешивается ровно на «открывающую» (▶) ячейку
   операции — в обеих строках шапки, в строках данных и в total-row'ах.
   Цвет намеренно полупрозрачно-чёрный (а не var(--display-border-strong)),
   чтобы линия читалась на TV, но не «резала» взгляд оператора и не
   создавала ощущения колонок-в-колонках. Толщину 2px подобрали под
   1080p TV: 1px на больших экранах теряется, 3+ — превращается
   в визуальный шов. На первой операции маршрута класс не ставится
   (см. `opIdx > 0` в `display-board.tsx`), потому что слева от неё
   и так уже стоят постоянные колонки со своим border'ом. */
.display-matrix__op-divider {
  border-left: 2px solid rgba(0, 0, 0, 0.15);
}
/* Дополнительные «потоковые» разделители: визуально режут матрицу
   на зоны маршрута —
       КРОЙ │ SEWING │ ОТК │ ВТО │ УПАКОВКА │ ГОТОВО.
   Линии живут на «крайних» колонках самих зон, чтобы не
   дублироваться с `display-matrix__op-divider` (тот рисуется только
   МЕЖДУ sewing-операциями) и не зависеть от наличия/количества
   sewing-операций в маршруте.

   - cut-divider — `border-right` на колонке КРОЙ. На стороне TSX
     этот класс снимается, если `sewingRoute.length === 0` (см.
     `flowDividerClass` в `display-board.tsx`): иначе бы рядом с
     border-left от qc-divider'а получилась бы видимая «двойная
     линия» 2 px + 2 px на стыке КРОЙ↔ОТК.
   - qc-divider — `border-left` на первой колонке зоны ОТК
     (▶-подколонка split-блока «ОТК»). Ставится всегда: даже без
     sewing-операций между КРОЙ и ОТК остаётся ровно одна
     полупрозрачная линия (cut-divider в этом случае выключен).
   - wto-divider — `border-left` на первой колонке зоны ВТО
     (▶-подколонка split-блока «ВТО») И на колонке `PACKING`
     (как «правая граница» зоны ВТО). Один и тот же класс используется
     с двух сторон, потому что цвет/толщина у обоих стыков одинаковые;
     `border-collapse: separate` гарантирует, что соседние ячейки
     с border-left/right в 2 px не складываются в визуальную «4 px
     двойную линию» — между ними всегда 1 px gap из cellspacing.

   Цвет и толщина намеренно совпадают с `display-matrix__op-divider`,
   чтобы все «потоковые» разделители читались как ОДНА визуальная
   система, а не как «более жирная» граница зон. Шопфлор-оператор
   таким образом получает полный визуальный поток
       КРОЙ │ SEWING │ ОТК │ ВТО │ УПАКОВКА
   ровно теми же 2-px полупрозрачными вертикалями, что и
   разделение sewing-операций между собой — никакого «секретного
   языка» классов, оператор сразу понимает «здесь начинается
   следующая зона маршрута». */
.display-matrix__cut-divider {
  border-right: 2px solid rgba(0, 0, 0, 0.15);
}
.display-matrix__qc-divider {
  border-left: 2px solid rgba(0, 0, 0, 0.15);
}
.display-matrix__wto-divider {
  border-left: 2px solid rgba(0, 0, 0, 0.15);
}
.display-visually-hidden {
  position: absolute;
  width: 1px;
  height: 1px;
  padding: 0;
  margin: -1px;
  overflow: hidden;
  clip: rect(0 0 0 0);
  white-space: nowrap;
  border: 0;
}
@media (max-width: 1199px) {
  .display-board {
    grid-template-columns: 1fr;
  }
}

.display-block {
  background: var(--display-bg-card);
  border: 1px solid var(--display-border);
  border-radius: 1rem;
  padding: 1rem 1.25rem;
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
  min-height: 0;
  overflow: hidden;
  box-shadow: var(--display-shadow);
}
.display-block__title {
  margin: 0;
  font-size: clamp(1rem, 1.2vw, 1.3rem);
  font-weight: 700;
  text-transform: uppercase;
  letter-spacing: 0.16em;
  color: var(--display-fg-muted);
  display: inline-flex;
  align-items: baseline;
  gap: 0.75rem;
}
.display-block__title-badge {
  font-size: 0.78em;
  font-weight: 700;
  letter-spacing: 0.04em;
  padding: 0.1rem 0.55rem;
  border-radius: 999px;
  background: var(--display-bg);
  color: var(--display-fg);
  border: 1px solid var(--display-border);
}
.display-block__title-sub {
  font-size: 0.75em;
  font-weight: 500;
  letter-spacing: 0.08em;
  text-transform: none;
  color: var(--display-fg-subtle);
}
.display-empty {
  font-size: clamp(1.1rem, 1.4vw, 1.5rem);
  font-weight: 600;
  color: var(--display-fg-subtle);
  text-align: center;
  /* Когда `display-empty` — единственный «контентный» ребёнок
     `.display-block` (нет матрицы / нет оборудования), он должен
     забрать всю остаточную высоту и центрировать сообщение, а не
     схлопываться в одну строку у заголовка. */
  flex: 1 1 auto;
  min-height: 0;
  display: flex;
  align-items: center;
  justify-content: center;
  padding: 1rem 0;
}

/* Production matrix -------------------------------------------------------- */
.display-matrix__scroll {
  /* Главный фикс TV/fullscreen geometry: scroll-область матрицы
     ОБЯЗАНА явно занимать остаток высоты внутри `.display-block` и
     иметь `min-height: 0`, иначе flex-item получает неявный
     `min-height: auto` = высоте таблицы → таблица не «помещается»
     в block, а сам block огорожен `overflow: hidden`, и на экране
     остаются только заголовки. */
  flex: 1 1 auto;
  min-height: 0;
  /* Явно ставим `display: block`: scroll-ancestor для sticky `<th>`
     должен быть устойчивым block-formatting контекстом, а не
     случайно унаследованным flex/grid (см. ниже про sticky). */
  display: block;
  overflow: auto;
  /* Изолируем repaint и контейн-границы: на TV WebKit это даёт
     более предсказуемый sticky без «прыгающих» заголовков. */
  contain: layout paint;
}
.display-matrix {
  width: 100%;
  border-collapse: separate;
  border-spacing: 0;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.display-matrix__th {
  position: sticky;
  top: 0;
  /* Непрозрачный фон + `background-clip: padding-box` обязательны для
     sticky `<th>`: иначе при скролле под ним просвечивают строки.
     `z-index: 2` поднимает sticky выше color-row (z-index: 1, см.
     ниже) — без этого блочный заголовок цвета может «налазить»
     поверх sticky на TV/WebKit. */
  background: var(--display-bg-soft);
  background-clip: padding-box;
  font-size: 0.85rem;
  text-transform: uppercase;
  letter-spacing: 0.1em;
  font-weight: 700;
  color: var(--display-fg-muted);
  padding: 0.65rem 0.6rem;
  text-align: center;
  border-bottom: 1px solid var(--display-border-strong);
  white-space: nowrap;
  z-index: 2;
}
.display-matrix__th--first {
  text-align: left;
  padding-left: 1rem;
}
.display-matrix__th-icon {
  display: inline-flex;
  width: 1.1rem;
  height: 1.1rem;
  vertical-align: -0.2rem;
  margin-right: 0.35rem;
  color: var(--display-fg-muted);
}
.display-matrix__th-icon svg {
  width: 100%;
  height: 100%;
}
.display-matrix__th--ok {
  color: var(--display-ok);
}
.display-matrix__th--ok .display-matrix__th-icon {
  color: var(--display-ok);
}
.display-matrix__th--ok-soft {
  color: var(--display-ok);
}
.display-matrix__th--accent {
  color: var(--display-accent);
}
.display-matrix__th--accent .display-matrix__th-icon {
  color: var(--display-accent);
}
.display-matrix__th--defect {
  color: var(--display-crit);
}
.display-matrix__th--defect .display-matrix__th-icon {
  color: var(--display-crit);
}
.display-matrix__cell {
  padding: 0.55rem 0.6rem;
  text-align: center;
  font-size: clamp(1.1rem, 1.4vw, 1.5rem);
  font-weight: 600;
  color: var(--display-fg);
  border-bottom: 1px solid var(--display-border);
  /*
   * Нужен только для ::after-оверлея flash-анимации (см.
   * `.display-cell--changed::after`). Без `position: relative` оверлей
   * с `position: absolute; inset: 0` ушёл бы к ближайшему позиционированному
   * предку (целому table'у/scroll-контейнеру) и красная заливка растянулась
   * бы по всей ширине экрана. Стоимость — 0: это чистый CSS-флаг,
   * никакой layout-работы он не вызывает.
   */
  position: relative;
}
/*
 * «Мигание красным» при изменении значения ячейки между двумя
 * успешными polling-snapshot'ами (см. `computeChangedCellKeys` и
 * `FlashCell` в display-board.tsx).
 *
 * Реализация через ::after-оверлей, а не через `background-color` на
 * самой td — намеренно. Прямая анимация bg перетёрла бы базовые
 * фоны total-row'ов (`.display-matrix__total-row` = #f8fafc,
 * `--grand` = var(--display-fg)) и в моменты 0%/100% keyframes
 * (когда альфа = 0) показывала бы прозрачный td поверх таблицы —
 * визуально это «отрубило» бы фон до окончания анимации.
 *
 * Оверлей же:
 *   - живёт ПОВЕРХ background'а ячейки (включая «итоговые» ряды),
 *     поэтому сам фон td не страдает;
 *   - имеет `pointer-events: none`, чтобы не мешать hover'у на
 *     `.display-matrix__row` и не «съедать» клики (актуально, если
 *     дисплей в будущем станет интерактивным);
 *   - содержимое td в normal-flow остаётся внизу под оверлеем;
 *     при alpha = 0.4 цифры читаются нормально на короткий промежуток
 *     1.2 c (две итерации × 0.6 c).
 *
 * Сама анимация навешана на ::after, который существует только при
 * добавленном классе `display-cell--changed`. Класс вычисляется на
 * каждом render'е из diff'а snapshot'ов (см. `FlashCell`), поэтому
 * никакого setState/setTimeout «внутри ячейки» нет — это чистый
 * css transition по факту изменения className.
 *
 * Перезапуск анимации на двух подряд изменениях одной и той же
 * ячейки обеспечивается ключом `key={'…|v=${value}'}` на td в
 * `FlashCell`: при смене value React пересоздаёт DOM-узел td
 * (а вместе с ним и ::after), и keyframes гарантированно играют
 * заново. На «спокойных» ячейках key стабилен → remount'ов нет.
 */
.display-cell--changed::after {
  content: '';
  position: absolute;
  inset: 0;
  pointer-events: none;
  background-color: rgba(255, 0, 0, 0);
  animation: display-cell-flash-red 0.6s ease-in-out 2;
}
@keyframes display-cell-flash-red {
  0% {
    background-color: rgba(255, 0, 0, 0);
  }
  50% {
    background-color: rgba(255, 0, 0, 0.4);
  }
  100% {
    background-color: rgba(255, 0, 0, 0);
  }
}
@media (prefers-reduced-motion: reduce) {
  /*
   * На системах с включённой опцией «уменьшить анимацию» (актуально
   * в первую очередь для пользователей с вестибулярными расстройствами,
   * но также часто включается на реальных TV/embedded WebView, чтобы
   * сэкономить GPU) выключаем flash полностью. Контракт продукта —
   * «мигание красным», без него ячейка просто молча обновится; это
   * лучше, чем игнорировать системные настройки доступности.
   */
  .display-cell--changed::after {
    animation: none;
  }
}
/*
 * Подсветка «узкого места» (bottleneck) в split-блоках матрицы.
 *
 * Логика выбора колонки — `detectBottlenecks` в display-board.tsx:
 * мы подсвечиваем НЕ ту операцию, у которой самый большой ✔-буфер,
 * а её СЛЕДУЮЩУЮ операцию — именно она тормозит и не успевает
 * разбирать накопленный буфер. Класс `display-matrix__bottleneck-col`
 * вешается на ВСЕ ячейки её столбца — и шапку, и обе подколонки
 * (▶ / ✔), и body-rows, и color-totals, и grand-totals, — чтобы
 * пульс шёл по всей вертикали столбца. Класс
 * `display-matrix__th--bottleneck` дополняет шапку (там же стоит
 * иконка ⚠).
 *
 * Цвет — мягкий coral (см. блок-комментарий у переменной
 * `--display-bottleneck-color`). Анимация намеренно медленная (2.4 c)
 * и низкоконтрастная (alpha 0.05 → 0.18), чтобы пульс «дышал», а
 * не «мигал» — это фоновая визуальная подсказка, она не должна
 * перетягивать внимание с чисел и flash'а изменений.
 */
.display-matrix__th--bottleneck {
  position: relative;
}
.display-matrix__bottleneck-mark {
  margin-left: 0.25rem;
  font-size: 0.85em;
  color: var(--display-bottleneck-color);
}
.display-matrix__bottleneck-col {
  background-color: rgba(251, 113, 133, 0.08);
  animation: display-bottleneck-pulse 2.4s ease-in-out infinite;
}
@keyframes display-bottleneck-pulse {
  0% {
    background-color: rgba(251, 113, 133, 0.05);
  }
  50% {
    background-color: rgba(251, 113, 133, 0.18);
  }
  100% {
    background-color: rgba(251, 113, 133, 0.05);
  }
}
@media (prefers-reduced-motion: reduce) {
  /*
   * Системы с включённой опцией «уменьшить анимацию» (включая
   * embedded TV/WebView, на которых её часто включают ради экономии
   * GPU): убираем сам пульс, но статический мягкий тон оставляем —
   * иначе оператор не увидит подсветку bottleneck'а вообще, а
   * именно она и есть основной продуктовый сигнал.
   */
  .display-matrix__bottleneck-col {
    animation: none;
    background-color: rgba(251, 113, 133, 0.12);
  }
}
.display-matrix__cell--ok {
  color: var(--display-ok);
  font-weight: 700;
}
.display-matrix__cell--ok-soft {
  color: var(--display-ok);
}
.display-matrix__cell--accent {
  color: var(--display-accent);
  font-weight: 700;
}
.display-matrix__cell--defect {
  color: var(--display-crit);
}
.display-matrix__row-label {
  text-align: left;
  padding: 0.55rem 0.6rem 0.55rem 1rem;
  font-size: clamp(0.95rem, 1.1vw, 1.15rem);
  font-weight: 600;
  color: var(--display-fg-muted);
  border-bottom: 1px solid var(--display-border);
  white-space: nowrap;
}
.display-matrix__row:hover .display-matrix__cell,
.display-matrix__row:hover .display-matrix__row-label {
  background: var(--display-bg-soft);
}
.display-matrix__color-row .display-matrix__color-label {
  text-align: left;
  padding: 0.65rem 0.6rem 0.55rem 1rem;
  font-size: 0.95rem;
  text-transform: uppercase;
  letter-spacing: 0.14em;
  font-weight: 700;
  color: var(--display-fg-muted);
  background: var(--display-bg-soft);
  border-top: 1px solid var(--display-border-strong);
  border-bottom: 1px solid var(--display-border);
}
.display-matrix__color-swatch {
  display: inline-block;
  width: 0.85rem;
  height: 0.85rem;
  border-radius: 0.2rem;
  margin-right: 0.55rem;
  vertical-align: -0.1rem;
  border: 1px solid var(--display-border-strong);
}
.display-matrix__color-swatch--black {
  background: #0f172a;
}
.display-matrix__color-swatch--white {
  background: #ffffff;
}
.display-matrix__total-row .display-matrix__cell,
.display-matrix__total-row .display-matrix__row-label {
  font-weight: 700;
  background: #f8fafc;
  border-bottom: 1px solid var(--display-border-strong);
}
.display-matrix__total-row--grand .display-matrix__cell,
.display-matrix__total-row--grand .display-matrix__row-label {
  background: var(--display-fg);
  color: #f8fafc;
  font-size: clamp(1.15rem, 1.5vw, 1.6rem);
  border-top: 2px solid var(--display-fg);
  border-bottom: 0;
}
.display-matrix__total-row--grand .display-matrix__cell--ok,
.display-matrix__total-row--grand .display-matrix__cell--ok-soft {
  color: #86efac;
}
.display-matrix__total-row--grand .display-matrix__cell--accent {
  color: #93c5fd;
}
.display-matrix__total-row--grand .display-matrix__cell--defect {
  color: #fecaca;
}

/* Equipment panel ----------------------------------------------------------
 *
 * Радикально упрощённый блок: только сетка маленьких плиток
 * `<номер> + цветной dot`. Никаких иконок, имён, размеров, ФИО,
 * текста статуса. Цель — уместить 30+ станков без скролла на TV
 * и считываться оператором за полсекунды.
 *
 * Контракт UI см. в `EquipmentPanel` (display-board.tsx).
 */
.display-equipment-grid {
  list-style: none;
  margin: 0;
  padding: 0;
  display: grid;
  /* 4–6 колонок в ряд: на ≥1080p ширина блока ~360–400px,
     `minmax(56px, 1fr)` даёт 5–6 квадратных плиток в строке. */
  grid-template-columns: repeat(auto-fill, minmax(56px, 1fr));
  gap: 0.4rem;
  /* Та же история, что и у `.display-matrix__scroll`: грид-плитки
     должны жить внутри собственной scroll-зоны и НЕ выталкивать
     соседей (totals) за пределы блока. */
  flex: 1 1 auto;
  min-height: 0;
  overflow-y: auto;
  align-content: flex-start;
}
.display-equipment-tile {
  position: relative;
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  gap: 0.2rem;
  aspect-ratio: 1 / 1;
  /* Минимальный padding — экран на 30+ станков не должен «дышать»
     лишними пикселями. */
  padding: 0.25rem 0.2rem;
  border-radius: 0.5rem;
  background: var(--display-bg-soft);
  border: 1px solid var(--display-border);
  color: var(--display-fg);
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  transition: none;
}
.display-equipment-tile__num {
  font-size: clamp(1rem, 1.4vw, 1.45rem);
  font-weight: 800;
  letter-spacing: 0.02em;
  line-height: 1;
}
/* Вторичный визуальный маркер «тип оборудования» внутри строки с
   номером (см. компонент `EquipmentKindIcon` в display-board.tsx).
   Иконка — тонкий line-SVG из `lucide-react` со `strokeWidth={1.5}`,
   рендерится как `<svg className="display-equipment-tile__icon" />`.
   Размер задаём в em (≈ 0.9 от `__num`) и приглушаем opacity до 0.65,
   чтобы иконка помогала «за полсекунды распознать тип станка», но
   НЕ перетягивала внимание с основного сигнала плитки (номер +
   актуальные размеры; статус читается цветом ВСЕЙ плитки, отдельный
   dot убран намеренно).
   margin-right даёт минимальный визуальный зазор между иконкой и
   цифрой без увеличения высоты плитки и без ломания компактного
   auto-fill грида на 30+ станков. */
.display-equipment-tile__icon {
  width: 0.9em;
  height: 0.9em;
  font-size: 0.9em;
  margin-right: 3px;
  opacity: 0.65;
  flex-shrink: 0;
  vertical-align: -0.1em;
}
/* Текущие размеры на станке. Намеренно компактные:
   отдельная маленькая строка между номером и цветным dot'ом —
   так оператор за полсекунды видит «01 / M / зелёный dot»
   и сразу понимает, что именно сейчас идёт на станке.
   Если размеров несколько ("M,L") — рендерим в одну строку
   через запятую без пробела (узкая плитка), длинное —
   обрезается ellipsis'ом, чтобы плитка не растягивалась. */
.display-equipment-tile__sizes {
  font-size: clamp(0.7rem, 0.85vw, 0.9rem);
  font-weight: 700;
  line-height: 1;
  letter-spacing: 0.02em;
  color: inherit;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
  max-width: 100%;
}
.display-equipment-tile--online {
  background: var(--display-ok-soft);
  border-color: #86efac;
  color: var(--display-ok);
}
.display-equipment-tile--online .display-equipment-tile__num {
  color: #14532d;
}
.display-equipment-tile--warning {
  background: var(--display-warn-soft);
  border-color: #fcd34d;
  color: var(--display-warn);
}
.display-equipment-tile--warning .display-equipment-tile__num {
  color: #78350f;
}
.display-equipment-tile--offline {
  background: var(--display-bg-card);
  border-color: var(--display-border);
  color: var(--display-fg-subtle);
}
.display-equipment-tile--offline .display-equipment-tile__num {
  color: var(--display-fg-subtle);
}

/* -----------------------------------------------------------------------
 * Кнопка «Мастер» на рабочих терминалах (mobile-first).
 *
 * Видимая, но не перекрывает основной scan-flow: floating bottom-right,
 * крупная (≥56px), уверенно нажимается большим пальцем. После нажатия —
 * меняется на «Мастер вызван» и блокируется (см. идемпотентность в
 * `MasterCallsService.create`). Цвет — тот же violet/blue, что и
 * пульс на display'е (`display-equipment-tile--master-call`), чтобы
 * у мастера и у швеи был один визуальный язык.
 * ----------------------------------------------------------------------- */
.call-master-wrap {
  pointer-events: none;
}
.call-master-btn {
  pointer-events: auto;
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.85rem 1.4rem;
  border: 0;
  border-radius: 999px;
  background: #4f46e5;
  color: #ffffff;
  font-size: 1rem;
  font-weight: 700;
  letter-spacing: 0.02em;
  cursor: pointer;
  box-shadow: 0 6px 14px rgba(79, 70, 229, 0.35);
  transition: transform 0.1s ease, background 0.15s ease;
}
.call-master-btn:hover:not(:disabled) {
  background: #4338ca;
}
.call-master-btn:active:not(:disabled) {
  transform: scale(0.97);
}
.call-master-btn:disabled {
  background: #6d6abf;
  cursor: default;
  opacity: 0.85;
}
.call-master-btn[data-state='called'] {
  background: #16a34a;
  box-shadow: 0 6px 14px rgba(22, 163, 74, 0.35);
}
.call-master-btn__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 1.4rem;
  height: 1.4rem;
  border-radius: 999px;
  background: rgba(255, 255, 255, 0.2);
  font-weight: 800;
  font-size: 0.95rem;
}
.call-master-btn__label {
  white-space: nowrap;
}
.call-master-btn__error {
  pointer-events: auto;
  margin: 0.4rem 0 0;
  padding: 0.4rem 0.6rem;
  background: #fef2f2;
  border: 1px solid #fecaca;
  border-radius: 0.4rem;
  color: #991b1b;
  font-size: 0.85rem;
  max-width: 18rem;
}

/* Floating вариант — для рабочих терминалов. Нижний-правый угол с
 * отступом для safe-area iOS. */
.call-master-btn--fab {
  position: fixed;
  right: 1rem;
  bottom: calc(env(safe-area-inset-bottom, 0px) + 1rem);
  z-index: 90;
  min-height: 56px;
  min-width: 56px;
}
@media (max-width: 480px) {
  .call-master-btn--fab {
    right: 0.75rem;
    bottom: calc(env(safe-area-inset-bottom, 0px) + 0.75rem);
  }
}

/* -----------------------------------------------------------------------
 * Кнопка «Мой QR-код» + модалка
 * (см. `apps/web/components/employees/employee-qr-button.tsx`,
 * `docs/screens.md §«Мой QR-код»`).
 *
 * Визуально — родственник `.call-master-btn`, но с нейтральной
 * тёмно-синей палитрой: мастер — оранжевый/фиолетовый акцент,
 * «Мой QR» — спокойная «ссылка на себя», чтобы две кнопки на одном
 * экране не конкурировали за внимание.
 * ----------------------------------------------------------------------- */
.employee-qr-btn {
  pointer-events: auto;
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.6rem 1rem;
  border: 0;
  border-radius: 999px;
  background: rgba(255, 255, 255, 0.14);
  color: #ffffff;
  font-size: 0.95rem;
  font-weight: 600;
  letter-spacing: 0.01em;
  cursor: pointer;
  transition: background 0.15s ease, transform 0.1s ease;
}
.employee-qr-btn:hover:not(:disabled) {
  background: rgba(255, 255, 255, 0.22);
}
.employee-qr-btn:active:not(:disabled) {
  transform: scale(0.97);
}
.employee-qr-btn__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 1.3rem;
  height: 1.3rem;
  border-radius: 6px;
  background: rgba(255, 255, 255, 0.22);
  font-weight: 800;
  font-size: 0.9rem;
}
.employee-qr-btn__label { white-space: nowrap; }

/* Inline-вариант на «светлых» карточках (например, `/master`). */
.employee-qr-btn--inline.employee-qr-btn {
  background: #e0e7ff;
  color: #3730a3;
}
.employee-qr-btn--inline.employee-qr-btn:hover:not(:disabled) {
  background: #c7d2fe;
}
.employee-qr-btn--inline.employee-qr-btn .employee-qr-btn__icon {
  background: #ffffff;
  color: #3730a3;
}

/* Внутри role-header (синяя шапка) оставляем белый полупрозрачный фон. */
.role-header .employee-qr-btn--inline.employee-qr-btn {
  background: rgba(255, 255, 255, 0.16);
  color: #ffffff;
}
.role-header .employee-qr-btn--inline.employee-qr-btn:hover:not(:disabled) {
  background: rgba(255, 255, 255, 0.24);
}
.role-header .employee-qr-btn--inline.employee-qr-btn .employee-qr-btn__icon {
  background: rgba(255, 255, 255, 0.24);
  color: #ffffff;
}

/* Floating-вариант — уровнем выше «Мастера», но левее, чтобы не
   перекрывал. Используется, когда кнопку надо видеть всегда. */
.employee-qr-btn--fab {
  position: fixed;
  left: 1rem;
  bottom: calc(env(safe-area-inset-bottom, 0px) + 1rem);
  z-index: 90;
  min-height: 48px;
  background: #1e293b;
  color: #ffffff;
  box-shadow: 0 6px 14px rgba(15, 23, 42, 0.35);
}
.employee-qr-btn--fab:hover:not(:disabled) { background: #0f172a; }
@media (max-width: 480px) {
  .employee-qr-btn--fab {
    left: 0.75rem;
    bottom: calc(env(safe-area-inset-bottom, 0px) + 0.75rem);
  }
}

/* Терминалы сотрудников (`/work`, `/qc`, `/wto`, `/packing`):
   «Мой QR-код» + «Мастер» закреплены СПРАВА вертикальным столбиком,
   а не FAB-ами в нижних углах (низ часто занят прилипшей панелью
   действий, напр. `.qc-card__sticky-actions`). Сверху-справа на
   каждом из этих экранов уже стоит три-точечное меню
   `.seamstress-actions` (top 0.4rem, 40×40), поэтому столбик
   начинается ПОД ним (top = safe-area + 0.4rem меню + 40px высота
   меню + 60px зазор). Скоуп — все четыре терминала, оборачиваются
   в layout'е раздела единым `.employee-toolbar`.

   Кнопки внутри столбика — icon-only 40×40 (минимум места при
   сохранении one-tap, точно как trigger меню `.seamstress-actions`).
   Подписи скрываем, имя кнопки даёт `aria-label` на самом `<button>`.
   Цветовая семантика «Мастера» (индиго → зелёный при
   `data-state='called'`) и disabled-стейт остаются как было — они и
   без текста сообщают статус.

   На `/master` (кабинет мастера) этот тулбар НЕ нужен: там нет
   `.seamstress-actions` для якоря, а у мастера и нет кнопки «Мастер».
   QR-кнопка там остаётся встроенной в страницу. */
.employee-toolbar {
  position: fixed;
  /* safe-area + 0.4rem (инсет меню) + 40px (высота trigger'а
     `.seamstress-actions__trigger`) + 80px зазор. */
  top: calc(env(safe-area-inset-top, 0px) + 0.4rem + 40px + 80px);
  /* Выравниваем по правому краю с `.seamstress-actions` (position:
     absolute, right: 0.4rem внутри `.work`), а не с краем вьюпорта.
     `.work` — это контент `.app-main` с padding 1.25rem (0.85rem на
     <=640px) и max-width 1280px по центру; поэтому к 0.4rem меню
     добавляем padding `.app-main` и зазор центрирования. */
  right: calc(max((100vw - 1280px) / 2, 0px) + 1.25rem + 0.4rem);
  z-index: 80;
  display: flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 0.5rem;
  pointer-events: none;
}
@media (max-width: 640px) {
  .employee-toolbar {
    right: calc(max((100vw - 1280px) / 2, 0px) + 0.85rem + 0.4rem);
  }
}
.employee-toolbar .call-master-btn--fab,
.employee-toolbar .employee-qr-btn--fab {
  position: static;
  inset: auto;
  pointer-events: auto;
}
.employee-toolbar .call-master-btn,
.employee-toolbar .employee-qr-btn {
  /* 40×40 — как `.seamstress-actions__trigger` (меню «⋯/Выйти»). */
  width: 40px;
  height: 40px;
  min-width: 40px;
  min-height: 40px;
  padding: 0;
  gap: 0;
  justify-content: center;
  border-radius: 50%;
}
.employee-toolbar .call-master-btn__label,
.employee-toolbar .employee-qr-btn__label {
  display: none;
}
.employee-toolbar .call-master-btn__icon,
.employee-toolbar .employee-qr-btn__icon {
  width: 1.4rem;
  height: 1.4rem;
  font-size: 0.95rem;
}

/* Карточка модалки «Мой QR» — вариант общего .qr-modal__card. */
.employee-qr-modal__card {
  max-width: 360px;
  gap: 0.9rem;
}
.employee-qr-modal__body {
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 0.9rem;
}
.employee-qr-modal__canvas {
  padding: 1rem;
  background: #ffffff;
  border-radius: 12px;
  box-shadow: 0 4px 12px rgba(15, 23, 42, 0.12);
  line-height: 0;
}
/* -----------------------------------------------------------------------
 * QrCodeView — единый frontend-рендер QR (см.
 * `apps/web/components/qr/qr-code-view.tsx`).
 *
 * Минимальные правила, чтобы SVG не схлопывался и не «терялся» на
 * белом фоне (которое уже произошло однажды и привело к hotfix'у —
 * см. `docs/qr-regression-recon.md`).
 * ----------------------------------------------------------------------- */
.qr-code-view {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  line-height: 0;
}
.qr-code-view svg {
  display: block;
  width: 100%;
  height: auto;
  max-width: 100%;
  /* Жёстко чёрный квадратик QR на белом фоне — иначе тёмная тема
     способна сделать «белое на белом» и сканер ничего не увидит. */
  color: #0f172a;
  background: #ffffff;
}
.qr-code-view--empty {
  min-width: 8rem;
  min-height: 8rem;
  padding: 1rem;
  border-radius: 12px;
  border: 1px dashed var(--color-border, #cbd5e1);
  color: var(--color-fg-muted, #64748b);
  font-size: 0.9rem;
  text-align: center;
  line-height: 1.3;
}
.employee-qr-modal__meta {
  text-align: center;
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
}
.employee-qr-modal__name {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 700;
  color: var(--color-fg, #0f172a);
}
.employee-qr-modal__role {
  margin: 0;
  font-size: 0.85rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  color: var(--color-fg-muted, #64748b);
}
.employee-qr-modal__retry {
  margin-top: 0.6rem;
}

/* -----------------------------------------------------------------------
 * Mobile-first экран `/master` (мастер цеха).
 *
 * Карточки вместо таблиц, большие кнопки, удобство одной рукой.
 * Цветовая палитра согласована с violet/blue пульсом display'а.
 * ----------------------------------------------------------------------- */
.master-page {
  max-width: 720px;
  margin: 0 auto;
  /* Top padding учитывает safe-area iOS (notch/dynamic-island), потому
     что глобальный `.app-header` для `/master` скрыт (см.
     `apps/web/components/app-header.tsx`, `hideForShopfloorMaster`)
     и страница рендерится «во весь экран». Bottom держим больше,
     чтобы sticky-кнопка «Сканировать QR» не наезжала на жест-бар. */
  padding:
    calc(env(safe-area-inset-top, 0px) + 1rem) 1rem
    calc(env(safe-area-inset-bottom, 0px) + 5rem);
  min-height: 100vh;
  display: flex;
  flex-direction: column;
  gap: 1rem;
  /* Якорь для абсолютного `.seamstress-actions` (меню «⋯ → Выйти»
     поверх угла синей `RoleHeaderCard`). */
  position: relative;
}

/* Fullscreen-режим `/master` (см. `apps/web/app/master/page.tsx`):
   глобальный `.app-header` для роли SHOPFLOOR_MASTER и для пути
   `/master` уже скрыт (см. `hideForShopfloorMaster` в
   `apps/web/components/app-header.tsx`), но `<main className="app-main">`
   из `apps/web/app/layout.tsx` остаётся с собственным padding'ом
   и `min-height: calc(100vh - 56px)` — это «съедает» полезную область
   на мобильном и портит ощущение терминала. Зачищаем его так же,
   как у `/shopfloor/display` (см. `body:has(.display-screen)`):
   `:has()` уже надёжно поддержан во всех современных браузерах,
   а `MobileNav` для SHOPFLOOR_MASTER скрыт через `singleWorkspace`
   (см. `apps/web/app/layout.tsx`), так что под `.master-page`
   ничего не остаётся, кроме самой страницы. */
body:has(.master-page) .app-main {
  padding: 0;
  max-width: none;
  min-height: 0;
}
/* Шапка мастера — общая синяя `RoleHeaderCard` + меню «⋯ → Выйти» в
   углу карты + «Мой QR-код» в боковом столбике. `.master-page` —
   полноэкранный контейнер (max-width 720px, свой `padding: 1rem`),
   поэтому абсолютный `.seamstress-actions` сдвигаем внутрь на величину
   паддинга (как в `.work` на остальных терминалах), а фиксированный
   `.employee-toolbar` выравниваем по правому краю 720px-колонки. */
.master-page .seamstress-actions {
  top: calc(env(safe-area-inset-top, 0px) + 1rem + 0.4rem);
  right: calc(1rem + 0.4rem);
}
.master-page .employee-toolbar {
  right: calc(max((100vw - 720px) / 2, 0px) + 1rem);
}
.master-page__primary {
  position: sticky;
  top: 0;
  z-index: 5;
  display: block;
  width: 100%;
  padding: 1rem;
  border: 0;
  border-radius: 0.8rem;
  background: #4f46e5;
  color: #fff;
  font-size: 1.1rem;
  font-weight: 700;
  cursor: pointer;
  box-shadow: 0 6px 14px rgba(79, 70, 229, 0.35);
}
.master-page__primary:active {
  transform: scale(0.99);
}
.master-page__scan-passport {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 0.5rem;
  width: 100%;
  min-height: 56px;
  padding: 0.85rem 1rem;
  border: 0;
  border-radius: 0.8rem;
  background: #0f766e;
  color: #fff;
  font-size: 1.05rem;
  font-weight: 700;
  cursor: pointer;
  box-shadow: 0 6px 14px rgba(15, 118, 110, 0.3);
}
.master-page__scan-passport:active {
  transform: scale(0.99);
}
.master-page__scan-passport:disabled {
  opacity: 0.7;
  cursor: default;
}
.master-page__empty {
  margin: 2rem auto;
  text-align: center;
  color: #4b5563;
}
.master-page__success {
  padding: 0.8rem 1rem;
  background: #ecfdf5;
  border: 1px solid #6ee7b7;
  border-radius: 0.6rem;
  color: #065f46;
  font-weight: 600;
  text-align: center;
}
.master-page__error {
  padding: 0.7rem 1rem;
  background: #fef2f2;
  border: 1px solid #fecaca;
  border-radius: 0.6rem;
  color: #991b1b;
}
.master-call-card {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  padding: 1rem;
  background: #ffffff;
  border: 1px solid #e5e7eb;
  border-left: 4px solid #6366f1;
  border-radius: 0.8rem;
  box-shadow: 0 2px 6px rgba(15, 23, 42, 0.05);
}
.master-call-card__top {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 0.5rem;
}
.master-call-card__name {
  margin: 0;
  font-size: 1.1rem;
  font-weight: 700;
  color: #1e1b4b;
}
.master-call-card__waiting {
  font-size: 0.9rem;
  font-weight: 700;
  color: #4338ca;
}
.master-call-card__meta {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  color: #374151;
  font-size: 0.95rem;
}
.master-call-card__meta-row span:first-child {
  color: #6b7280;
  font-size: 0.8rem;
  text-transform: uppercase;
  letter-spacing: 0.05em;
  margin-right: 0.4rem;
}
/* Stage 2 «Действия мастера»: блок «Действия с кроем» в карточке вызова.
 * Список паспортов — теперь не chip'ы, а развёрнутые карточки с метаданными
 * (qtyCut, операция, статус, ячейка) и кнопкой «Действия» — bottom-sheet.
 * Mobile-first: вертикальный stack, тапаемая кнопка ≥56px высотой. */
.master-call-card__passports-block {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  padding-top: 0.4rem;
  border-top: 1px dashed #e5e7eb;
}
.master-call-card__passports-title {
  margin: 0;
  font-size: 0.85rem;
  text-transform: uppercase;
  letter-spacing: 0.05em;
  color: #6b7280;
}
.master-call-card__passports {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  margin: 0;
  padding: 0;
  list-style: none;
}
.master-call-card__passport {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
  padding: 0.6rem 0.75rem;
  background: #eef2ff;
  border: 1px solid #c7d2fe;
  border-radius: 0.6rem;
  font-size: 0.9rem;
  color: #1e1b4b;
}
.master-call-card__passport-main {
  display: flex;
  flex-wrap: wrap;
  gap: 0.35rem;
  align-items: baseline;
}
.master-call-card__passport-main strong {
  font-size: 1rem;
}
.master-call-card__passport-meta {
  display: flex;
  flex-wrap: wrap;
  gap: 0.3rem;
  font-size: 0.8rem;
  color: #4338ca;
}
.master-call-card__passport-actions {
  align-self: stretch;
  min-height: 44px;
  padding: 0.55rem 0.8rem;
  border: 0;
  border-radius: 0.5rem;
  background: #4f46e5;
  color: #fff;
  font-weight: 700;
  font-size: 0.95rem;
  cursor: pointer;
}
.master-call-card__passport-actions:active {
  transform: scale(0.99);
}

/* Stage 2 «Действия мастера» — mobile bottom-sheet выбора и подтверждения
 * действия. Полупрозрачный backdrop, карточка прижата вниз, контент
 * скроллится при необходимости (длинный список шагов маршрута). */
.master-actions-sheet {
  position: fixed;
  inset: 0;
  background: rgba(15, 23, 42, 0.55);
  display: flex;
  align-items: flex-end;
  justify-content: center;
  z-index: 100;
}
.master-actions-sheet__card {
  width: min(100%, 520px);
  /* `vh`-fallback для Safari < 15.4 (см. `.auth-screen`). */
  max-height: 92vh;
  max-height: 92dvh;
  overflow-y: auto;
  background: #fff;
  border-top-left-radius: 1rem;
  border-top-right-radius: 1rem;
  padding: 1rem 1rem 1.2rem;
  box-shadow: 0 -8px 24px rgba(15, 23, 42, 0.25);
}
.master-actions-sheet__header {
  display: flex;
  gap: 0.5rem;
  align-items: flex-start;
  justify-content: space-between;
  margin-bottom: 0.8rem;
}
.master-actions-sheet__title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 700;
}
.master-actions-sheet__subtitle {
  margin: 0.15rem 0 0;
  color: #374151;
  font-size: 0.9rem;
}
.master-actions-sheet__meta {
  margin: 0.2rem 0 0;
  color: #4b5563;
  font-size: 0.85rem;
}
.master-actions-sheet__close {
  background: transparent;
  border: 0;
  font-size: 1.6rem;
  line-height: 1;
  color: #6b7280;
  cursor: pointer;
  padding: 0.25rem 0.5rem;
}
.master-actions-sheet__menu {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.master-actions-sheet__menu-item {
  display: flex;
  flex-direction: column;
  align-items: flex-start;
  gap: 0.2rem;
  width: 100%;
  min-height: 56px;
  padding: 0.7rem 0.9rem;
  border: 1px solid #e5e7eb;
  border-radius: 0.6rem;
  background: #f9fafb;
  text-align: left;
  cursor: pointer;
}
.master-actions-sheet__menu-label {
  font-weight: 700;
  font-size: 1rem;
  color: #111827;
}
.master-actions-sheet__menu-hint {
  font-size: 0.85rem;
  color: #4b5563;
}
.master-actions-sheet__body {
  display: flex;
  flex-direction: column;
  gap: 0.7rem;
}
.master-actions-sheet__back {
  align-self: flex-start;
  background: transparent;
  border: 0;
  color: #4338ca;
  font-size: 0.9rem;
  cursor: pointer;
  padding: 0.25rem 0;
}
.master-actions-sheet__action-title {
  margin: 0;
  font-size: 1rem;
  font-weight: 700;
}
.master-actions-sheet__action-hint {
  margin: 0;
  color: #4b5563;
  font-size: 0.85rem;
}
.master-actions-sheet__field {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
}
.master-actions-sheet__label {
  font-size: 0.85rem;
  font-weight: 600;
  color: #374151;
}
.master-actions-sheet__required {
  color: #dc2626;
}
.master-actions-sheet__row {
  display: flex;
  gap: 0.4rem;
  align-items: stretch;
}
.master-actions-sheet__input,
.master-actions-sheet__textarea {
  flex: 1 1;
  min-height: 44px;
  padding: 0.55rem 0.7rem;
  border: 1px solid #d1d5db;
  border-radius: 0.5rem;
  font-size: 0.95rem;
  background: #fff;
}
.master-actions-sheet__textarea {
  min-height: 80px;
  resize: vertical;
}
.master-actions-sheet__scan {
  min-height: 44px;
  padding: 0.55rem 0.8rem;
  border: 0;
  border-radius: 0.5rem;
  background: #4338ca;
  color: #fff;
  font-weight: 600;
  cursor: pointer;
}
.master-actions-sheet__steps {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
}
.master-actions-sheet__step {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.55rem 0.7rem;
  border: 1px solid #e5e7eb;
  border-radius: 0.5rem;
  cursor: pointer;
  background: #f9fafb;
}
.master-actions-sheet__step--active {
  border-color: #4f46e5;
  background: #eef2ff;
}
.master-actions-sheet__placement {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
  margin-bottom: 0.4rem;
}
.master-actions-sheet__placement-option {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.45rem 0.6rem;
  border: 1px solid #e5e7eb;
  border-radius: 0.5rem;
  cursor: pointer;
  background: #f9fafb;
  font-size: 0.9rem;
}
.master-actions-sheet__error {
  margin: 0;
  color: #b91c1c;
  font-size: 0.85rem;
}
/* «Посмотреть историю паспорта» — read-only обзор в том же sheet'е.
   Высоту history-листа ограничиваем и даём вертикальный скролл, чтобы
   на мобильном экране история не выталкивала кнопку «Назад» за viewport
   (у P-20260505-0191, например, до 7 событий — упирается в потолок). */
.master-actions-sheet__menu-item--history {
  border-color: #c7d2fe;
  background: #eef2ff;
}
/* ОТК-действия мастера («зафиксировать брак» / «вернуть на доработку»)
   — тёплый акцент, чтобы визуально отделить контроль качества от
   правок владельца/маршрута ниже. */
.master-actions-sheet__menu-item--qc {
  border-color: #fcd9b6;
  background: #fff7ed;
}
.master-actions-sheet__history-status {
  margin: 0;
  color: #4b5563;
  font-size: 0.85rem;
}
.master-actions-sheet__history {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  max-height: 60vh;
  overflow-y: auto;
  padding-right: 0.2rem;
}
.master-actions-sheet__history-item {
  display: grid;
  grid-template-columns: minmax(8.5rem, auto) 1fr;
  gap: 0.6rem;
  padding: 0.55rem 0.7rem;
  border: 1px solid #e5e7eb;
  border-radius: 0.5rem;
  background: #f9fafb;
}
.master-actions-sheet__history-item--manual {
  border-color: #fde68a;
  background: #fffbeb;
}
.master-actions-sheet__history-time {
  color: #4b5563;
  font-size: 0.85rem;
  white-space: nowrap;
}
.master-actions-sheet__history-main {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
}
.master-actions-sheet__history-type {
  font-weight: 700;
  color: #111827;
  font-size: 0.95rem;
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem;
  align-items: baseline;
}
.master-actions-sheet__history-manual {
  font-weight: 500;
  color: #92400e;
  font-size: 0.8rem;
}
.master-actions-sheet__history-meta {
  display: flex;
  flex-wrap: wrap;
  gap: 0.6rem;
  color: #4b5563;
  font-size: 0.85rem;
}
.master-actions-sheet__confirm {
  min-height: 56px;
  margin-top: 0.4rem;
  padding: 0.85rem 1rem;
  border: 0;
  border-radius: 0.6rem;
  background: #16a34a;
  color: #fff;
  font-size: 1rem;
  font-weight: 700;
  cursor: pointer;
}
.master-actions-sheet__confirm:disabled {
  background: #9ca3af;
  cursor: default;
}

/* Stage 3 «Мастер цеха» — карточка «Ограничения выдачи кроя» в /master.
 * Mobile-first: крупные кнопки, компактный summary в виде <dl>. */
.master-cut-release-card {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  padding: 0.9rem 1rem;
  margin-bottom: 0.8rem;
  background: #fff;
  border: 1px solid #e5e7eb;
  border-radius: 0.8rem;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04);
}
.master-cut-release-card__header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.5rem;
}
.master-cut-release-card__title {
  margin: 0;
  font-size: 1rem;
  font-weight: 700;
  color: #111827;
}
.master-cut-release-card__badge {
  display: inline-flex;
  align-items: center;
  padding: 0.15rem 0.55rem;
  border-radius: 999px;
  background: #dcfce7;
  color: #166534;
  font-size: 0.75rem;
  font-weight: 700;
  text-transform: uppercase;
  letter-spacing: 0.05em;
}
.master-cut-release-card__empty {
  margin: 0;
  color: #4b5563;
  font-size: 0.9rem;
}
.master-cut-release-card__summary {
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
  margin: 0;
}
.master-cut-release-card__row {
  display: flex;
  justify-content: space-between;
  gap: 0.5rem;
  font-size: 0.95rem;
}
.master-cut-release-card__row > dt {
  color: #4b5563;
  font-weight: 500;
  margin: 0;
}
.master-cut-release-card__row > dd {
  margin: 0;
  color: #111827;
  font-weight: 600;
}
.master-cut-release-card__actions {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.master-cut-release-card__primary,
.master-cut-release-card__secondary {
  width: 100%;
  min-height: 56px;
  padding: 0.85rem 1rem;
  border: 0;
  border-radius: 0.6rem;
  font-size: 1rem;
  font-weight: 700;
  cursor: pointer;
}
.master-cut-release-card__primary {
  background: #4f46e5;
  color: #fff;
}
.master-cut-release-card__secondary {
  background: #fff;
  border: 1px solid #d1d5db;
  color: #b91c1c;
}
.master-cut-release-card__primary:disabled,
.master-cut-release-card__secondary:disabled {
  opacity: 0.7;
  cursor: default;
}

.master-call-card__actions {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.master-call-card__resolve {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 0.5rem;
  width: 100%;
  min-height: 56px;
  padding: 0.85rem 1rem;
  border: 0;
  border-radius: 0.6rem;
  background: #4f46e5;
  color: #fff;
  font-size: 1rem;
  font-weight: 700;
  cursor: pointer;
}
.master-call-card__resolve:active {
  transform: scale(0.99);
}
.master-call-card__resolve:disabled {
  opacity: 0.7;
  cursor: default;
}
.master-call-card__resolve-manual {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 0.5rem;
  width: 100%;
  min-height: 48px;
  padding: 0.7rem 1rem;
  border: 1px solid #93c5fd;
  border-radius: 0.6rem;
  background: #eff6ff;
  color: #1e3a8a;
  font-size: 0.95rem;
  font-weight: 700;
  cursor: pointer;
  transition: background 0.15s, border-color 0.15s;
}
.master-call-card__resolve-manual:hover {
  background: #dbeafe;
  border-color: #60a5fa;
}
.master-call-card__resolve-manual:active {
  transform: scale(0.99);
}
.master-call-card__resolve-manual:disabled {
  opacity: 0.6;
  cursor: default;
}
.master-archive__resolved-badge {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  padding: 0.3rem 0.6rem;
  border: 1px solid #6ee7b7;
  border-radius: 0.5rem;
  background: #ecfdf5;
  color: #065f46;
  font-size: 0.8rem;
  font-weight: 700;
  white-space: nowrap;
  flex-shrink: 0;
}

/* -----------------------------------------------------------------------
 * Архив закрытых вызовов на /master.
 * Сворачиваемая секция под активной очередью; визуально приглушённая,
 * чтобы не отвлекать от основного потока «открытых вызовов».
 * --------------------------------------------------------------------- */
.master-archive {
  margin-top: 1rem;
  border: 1px solid #e5e7eb;
  border-radius: 0.6rem;
  background: #f9fafb;
  overflow: hidden;
}
.master-archive__toggle {
  display: flex;
  align-items: center;
  justify-content: space-between;
  width: 100%;
  padding: 0.75rem 1rem;
  background: transparent;
  border: 0;
  font-size: 0.95rem;
  font-weight: 700;
  color: #374151;
  cursor: pointer;
}
.master-archive__chevron {
  font-size: 0.9rem;
  color: #6b7280;
}
.master-archive__empty {
  padding: 0.8rem 1rem;
  color: #6b7280;
  font-size: 0.9rem;
  border-top: 1px solid #e5e7eb;
}
.master-archive__list {
  list-style: none;
  margin: 0;
  padding: 0;
  border-top: 1px solid #e5e7eb;
}
.master-archive__item {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.75rem;
  padding: 0.65rem 1rem;
  border-bottom: 1px solid #f1f5f9;
}
.master-archive__item:last-child {
  border-bottom: 0;
}
.master-archive__item-main {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  min-width: 0;
}
.master-archive__item-main strong {
  color: #111827;
  font-size: 0.95rem;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}
.master-archive__item-meta {
  color: #6b7280;
  font-size: 0.8rem;
}
.master-archive__item-time {
  color: #4b5563;
  font-size: 0.8rem;
  white-space: nowrap;
  flex-shrink: 0;
}

/* -----------------------------------------------------------------------
 * Master call pulse — мягкий violet/blue индикатор «зовут мастера».
 *
 * Источник истины — `apps/api/src/modules/master-calls/*` +
 * `ShopfloorEquipmentStatusDto.hasOpenMasterCall`. UI на
 * `/shopfloor/display` навешивает класс `display-equipment-tile--master-call`
 * на ту плитку, у которой `hasOpenMasterCall === true`.
 *
 * Принципиально отдельный класс от bottleneck coral pulse
 * (`.display-matrix__bottleneck-col`): bottleneck — это «накопился
 * буфер, узкое место маршрута», аварийный coral; master-call —
 * «рабочий просит мастера», спокойный сигнал внимания. Цвет
 * сознательно не красный, чтобы оператор не путал две оси проблем.
 * Пульс мягкий и медленный (2.4 c) — на TV не отвлекает, но
 * заметен с расстояния.
 * ----------------------------------------------------------------------- */
.display-equipment-tile--master-call {
  position: relative;
  border-color: #818cf8;
  box-shadow: 0 0 0 0 rgba(99, 102, 241, 0.55);
  animation: display-equipment-tile-master-call-pulse 2.4s ease-in-out infinite;
}
.display-equipment-tile--master-call::after {
  content: '';
  position: absolute;
  top: 0.4rem;
  right: 0.4rem;
  width: 0.7rem;
  height: 0.7rem;
  border-radius: 999px;
  background: #6366f1;
  box-shadow: 0 0 0 3px rgba(99, 102, 241, 0.25);
}
@keyframes display-equipment-tile-master-call-pulse {
  0%, 100% {
    box-shadow: 0 0 0 0 rgba(99, 102, 241, 0.55);
  }
  50% {
    box-shadow: 0 0 0 8px rgba(99, 102, 241, 0);
  }
}
@media (prefers-reduced-motion: reduce) {
  .display-equipment-tile--master-call {
    animation: none;
    box-shadow: 0 0 0 4px rgba(99, 102, 241, 0.35);
  }
}

/* Orphan master-calls block (вызовы без `equipmentId`). */
.display-orphan-calls {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
  margin-top: 0.6rem;
  padding: 0.6rem 0.8rem;
  background: rgba(99, 102, 241, 0.10);
  border: 1px solid #818cf8;
  border-radius: 0.6rem;
  color: var(--display-fg, inherit);
}
.display-orphan-calls__title {
  font-size: 0.85rem;
  font-weight: 700;
  text-transform: uppercase;
  letter-spacing: 0.05em;
  color: #4338ca;
  margin: 0;
}
.display-orphan-calls__list {
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem;
  margin: 0;
  padding: 0;
  list-style: none;
}
.display-orphan-calls__item {
  display: inline-flex;
  align-items: baseline;
  gap: 0.4rem;
  padding: 0.25rem 0.6rem;
  background: #ffffff;
  border: 1px solid #c7d2fe;
  border-radius: 999px;
  font-size: 0.9rem;
  color: #1e1b4b;
}
.display-orphan-calls__waiting {
  font-size: 0.8rem;
  color: #4338ca;
}
/* Компактный totals-чип под сеткой: только цвет + цифра, без подписей.
   Оставлен для оператора как «один взгляд — сколько онлайн/простой/оффлайн». */
.display-equipment__totals {
  display: flex;
  justify-content: flex-start;
  gap: 0.6rem;
  padding-top: 0.4rem;
  border-top: 1px solid var(--display-border);
  font-size: 0.85rem;
  color: var(--display-fg-muted);
}
.display-equipment__total {
  display: inline-flex;
  align-items: baseline;
  gap: 0.25rem;
}
.display-equipment__total::before {
  content: '';
  display: inline-block;
  width: 0.5rem;
  height: 0.5rem;
  border-radius: 999px;
  background: currentColor;
  margin-right: 0.3rem;
  vertical-align: 0.05rem;
}
.display-equipment__total strong {
  font-size: 1.1rem;
  font-weight: 800;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.display-equipment__total--online {
  color: var(--display-ok);
}
.display-equipment__total--warning {
  color: var(--display-warn);
}
.display-equipment__total--offline {
  color: var(--display-crit);
}

/* Скрываем основной layout (`.app-main` padding) под этой страницей —
   `.display-screen` уже позиционируется `fixed; inset: 0`. Дополнительно
   убираем `max-width` и `min-height`, чтобы под fixed-обёрткой не
   возникало случайного scrollbar'а у `<body>` на TV/embedded WebView,
   где `:has()` поддерживается, но layout перезаписывается агрессивнее.
   На браузерах без `:has()` экран всё равно работает: `.display-screen`
   накрывает страницу `position: fixed` и не зависит от `.app-main`. */
body:has(.display-screen) .app-main {
  padding: 0;
  max-width: none;
  min-height: 0;
}
body:has(.display-screen) {
  /* Страховка от случайного скролла body на TV: вся геометрия
     дисплея сидит внутри `.display-screen` (fixed) и сама управляет
     своими внутренними scroll-зонами. */
  overflow: hidden;
}

/* TV / large-display layer ------------------------------------------------- */
/* Цели:
     - читаемость с расстояния 3-7 м на 44" мониторе (1920×1080 fullscreen);
     - KPI остаётся в одну строку из 8 карточек (никакого wrap'а в 4×2);
     - board остаётся двухколоночным (production + equipment);
     - матрица и плитки оборудования — крупнее, но НЕ ломают chain
       overflow (только font-size / padding / icon-size, layout цепочка
       выше остаётся прежней).
   Брейкпоинт `min-width: 1600px` намеренно выше десктопного (≥1199),
   чтобы laptop/desktop-вид (1280-1440) не перерастал в TV-режим.
   Все правила скоупаны под `.display-screen` — на остальные страницы
   приложения этот слой не влияет. */
@media (min-width: 1600px) {
  .display-screen {
    padding: 1.5rem 1.75rem 1.75rem;
    gap: 1.25rem;
  }
  .display-screen__brand {
    font-size: 1.8rem;
  }
  .display-screen__clock {
    font-size: 4rem;
  }
  .display-screen__meta {
    font-size: 1.1rem;
  }
  .display-kpi {
    gap: 1rem;
  }
  .display-kpi__card {
    padding: 1rem 1.15rem;
  }
  .display-kpi__value {
    font-size: 2.8rem;
  }
  .display-kpi__label {
    font-size: 0.95rem;
  }
  .display-block__title {
    font-size: 1.35rem;
  }
  .display-matrix__th {
    font-size: 1rem;
    padding: 0.85rem 0.7rem;
  }
  /* На TV первая строка шапки выше (font-size 1rem + padding 0.85rem),
     значит sticky подколонок должен прилипать к её низу — иначе
     второй ряд накладывается на первый и закрывает имя операции. */
  .display-matrix__th--sub {
    top: 2.85rem;
    font-size: 1.1rem;
    padding: 0.4rem 0.5rem;
  }
  .display-matrix__th--sub .display-matrix__th-dir {
    font-size: 1.25rem;
  }
  .display-matrix__cell {
    font-size: 1.7rem;
    padding: 0.7rem 0.7rem;
  }
  .display-matrix__row-label {
    font-size: 1.25rem;
    padding: 0.7rem 0.7rem 0.7rem 1.1rem;
  }
  .display-matrix__total-row--grand .display-matrix__cell,
  .display-matrix__total-row--grand .display-matrix__row-label {
    font-size: 1.85rem;
  }
  .display-equipment-grid {
    grid-template-columns: repeat(auto-fill, minmax(72px, 1fr));
    gap: 0.5rem;
  }
  .display-equipment-tile__num {
    font-size: 1.55rem;
  }
  .display-equipment-tile__sizes {
    font-size: 0.95rem;
  }
  .display-equipment__totals {
    font-size: 1rem;
  }
  .display-equipment__total strong {
    font-size: 1.3rem;
  }
}

/* 4K / большие панели (≥ 2400px). Только дополнительный апскейл шрифтов
   и плиток; layout остаётся таким же двухколоночным. */
@media (min-width: 2400px) {
  .display-screen__clock {
    font-size: 5rem;
  }
  .display-kpi__value {
    font-size: 3.4rem;
  }
  .display-matrix__cell {
    font-size: 2rem;
  }
  .display-matrix__total-row--grand .display-matrix__cell,
  .display-matrix__total-row--grand .display-matrix__row-label {
    font-size: 2.2rem;
  }
  .display-equipment-grid {
    grid-template-columns: repeat(auto-fill, minmax(88px, 1fr));
  }
  .display-equipment-tile__num {
    font-size: 1.85rem;
  }
  .display-equipment-tile__sizes {
    font-size: 1.1rem;
  }
}

/* =============================================================================
   Admin UI Polish (Шаг — Admin UI Polish)
   ----------------------------------------------------------------------------
   Светлая, человекоориентированная админка. Токены и компоненты живут
   рядом со старыми (`.admin-overview`, `.data-table`, …) — старый UI
   продолжает работать, новые страницы используют префикс `.admin-*`.

   Источник истины — TЗ «Admin UI Polish». Не путать с `--color-bg` /
   `.card`: эти токены и классы намеренно не пересекаются, чтобы
   страницы вне `/admin` (master/shopfloor/work/orders) визуально не
   изменились.
   ============================================================================= */

:root {
  --admin-bg: #f8fafc;
  --admin-surface: #ffffff;
  --admin-soft: #f1f5f9;
  --admin-border: #e2e8f0;
  --admin-text: #0f172a;
  --admin-muted: #64748b;

  --admin-primary: #2563eb;
  --admin-primary-soft: #dbeafe;
  --admin-primary-fg: #1e3a8a;

  --admin-success: #16a34a;
  --admin-success-soft: #dcfce7;
  --admin-success-fg: #166534;

  --admin-warning: #b45309;
  --admin-warning-soft: #fef3c7;
  --admin-warning-fg: #92400e;

  --admin-danger: #dc2626;
  --admin-danger-soft: #fee2e2;
  --admin-danger-fg: #991b1b;

  --admin-radius-card: 20px;
  --admin-radius-pill: 999px;

  --admin-shadow-soft: 0 1px 2px rgba(15, 23, 42, 0.04),
    0 4px 14px rgba(15, 23, 42, 0.04);
}

/* ---------------------------------------------------------------------------
   Admin shell — фон страницы и общий «воздух».
   --------------------------------------------------------------------------- */

.admin-shell {
  background: var(--admin-bg);
  min-height: 100%;
  padding: 1.5rem 1.25rem 3rem;
  display: grid;
  gap: 1.5rem;
  color: var(--admin-text);
}

.admin-shell__header {
  display: flex;
  flex-wrap: wrap;
  gap: 1rem;
  justify-content: space-between;
  align-items: flex-end;
}

.admin-shell__title {
  margin: 0;
  font-size: 1.65rem;
  font-weight: 700;
  letter-spacing: -0.01em;
  color: var(--admin-text);
  display: flex;
  align-items: center;
  gap: 0.6rem;
}

.admin-shell__subtitle {
  margin: 0.4rem 0 0;
  color: var(--admin-muted);
  font-size: 0.95rem;
  line-height: 1.5;
  max-width: 60ch;
}

.admin-shell__actions {
  display: flex;
  gap: 0.5rem;
  flex-wrap: wrap;
}

/* ---------------------------------------------------------------------------
   Admin card — основной строительный блок.
   --------------------------------------------------------------------------- */

.admin-card {
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: var(--admin-radius-card);
  padding: 1.25rem 1.4rem;
  box-shadow: var(--admin-shadow-soft);
  display: grid;
  gap: 0.85rem;
}

.admin-card--compact { padding: 1rem 1.15rem; }

.admin-card--clickable {
  text-decoration: none;
  color: inherit;
  transition: transform 0.12s ease, box-shadow 0.12s ease,
    border-color 0.12s ease;
  cursor: pointer;
}
.admin-card--clickable:hover {
  border-color: #cfd8e3;
  box-shadow: 0 6px 20px rgba(15, 23, 42, 0.08);
  transform: translateY(-1px);
  color: inherit;
}
.admin-card--clickable:focus-visible {
  outline: 2px solid var(--admin-primary);
  outline-offset: 2px;
}

/* ---------------------------------------------------------------------------
   Section header — h2 + опциональные actions/hint.
   --------------------------------------------------------------------------- */

.admin-section {
  display: grid;
  gap: 0.85rem;
}

.admin-section-header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 1rem;
  flex-wrap: wrap;
}

.admin-section-header__title {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  margin: 0;
  font-size: 1.1rem;
  font-weight: 600;
  color: var(--admin-text);
}

.admin-section-header__hint {
  color: var(--admin-muted);
  font-size: 0.88rem;
}

.admin-section-header__actions {
  display: flex;
  gap: 0.5rem;
  flex-wrap: wrap;
}

/* ---------------------------------------------------------------------------
   Admin table — лёгкая таблица без рамок-сетки.
   --------------------------------------------------------------------------- */

.admin-table-wrap {
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: var(--admin-radius-card);
  overflow: hidden;
  box-shadow: var(--admin-shadow-soft);
}

.admin-table {
  width: 100%;
  border-collapse: separate;
  border-spacing: 0;
  font-size: 0.95rem;
}

.admin-table thead th {
  text-align: left;
  font-weight: 600;
  font-size: 0.78rem;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--admin-muted);
  padding: 0.85rem 1rem;
  background: var(--admin-soft);
  border-bottom: 1px solid var(--admin-border);
}

.admin-table tbody td {
  padding: 0.95rem 1rem;
  border-top: 1px solid var(--admin-border);
  vertical-align: middle;
  color: var(--admin-text);
}

.admin-table tbody tr:first-child td { border-top: 0; }

.admin-table tbody tr:hover td {
  background: var(--admin-soft);
}

.admin-table__primary {
  font-weight: 600;
  color: var(--admin-text);
}

.admin-table__hint {
  color: var(--admin-muted);
  font-size: 0.85rem;
  margin-top: 0.15rem;
}

.admin-table__actions {
  text-align: right;
  white-space: nowrap;
}

.admin-table__action-link {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  color: var(--admin-primary);
  text-decoration: none;
  font-weight: 500;
}
.admin-table__action-link:hover { text-decoration: underline; }
.admin-table__action-link--danger { color: var(--admin-danger); }

/* ---------------------------------------------------------------------------
 * Объединённая матрица отчёта «Себестоимость» (operation-size-matrix.tsx).
 * Компактная, на всю ширину карточки; при нехватке места — горизонтальный
 * скролл, а первая колонка «Операция / Сотрудник» прилипает слева.
 * Отдельно от `.admin-table`, чтобы не наследовать крупный padding и
 * мобильный card-reflow (он ломает матрицу).
 * ------------------------------------------------------------------------- */
.cost-matrix-wrap {
  overflow-x: auto;
  border: 1px solid var(--admin-border);
  border-radius: var(--admin-radius-card);
  background: var(--admin-surface);
}
.cost-matrix {
  width: 100%;
  border-collapse: separate;
  border-spacing: 0;
  font-size: 0.78rem;
}
.cost-matrix thead th {
  text-align: left;
  font-weight: 600;
  font-size: 0.7rem;
  letter-spacing: 0.02em;
  text-transform: uppercase;
  color: var(--admin-muted);
  padding: 0.45rem 0.55rem;
  background: var(--admin-soft);
  border-bottom: 1px solid var(--admin-border);
  white-space: nowrap;
}
.cost-matrix tbody td,
.cost-matrix tfoot td {
  padding: 0.4rem 0.55rem;
  border-top: 1px solid var(--admin-border);
  vertical-align: middle;
  color: var(--admin-text);
}
.cost-matrix tbody tr:first-child td { border-top: 0; }
.cost-matrix tfoot td {
  background: var(--admin-soft);
  border-top: 2px solid var(--admin-border);
}
.cost-matrix tbody tr:hover td { background: var(--admin-soft); }
.cost-matrix__num { text-align: right; white-space: nowrap; }
.cost-matrix__sticky {
  position: sticky;
  left: 0;
  z-index: 1;
  background: var(--admin-surface);
  border-right: 1px solid var(--admin-border);
}
.cost-matrix thead th.cost-matrix__sticky { z-index: 2; background: var(--admin-soft); }
.cost-matrix tfoot td.cost-matrix__sticky { background: var(--admin-soft); }
.cost-matrix tbody tr:hover td.cost-matrix__sticky { background: var(--admin-soft); }

/* Поисковый combobox фильтров отчёта (filter-combobox.tsx). */
.cost-combo { position: relative; }
.cost-combo__control {
  display: flex;
  align-items: center;
  gap: 4px;
}
.cost-combo__control input[type='text'] {
  padding: 0.6rem 0.75rem;
  border-radius: 10px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font: inherit;
  width: 100%;
  transition: border-color 160ms ease, box-shadow 160ms ease;
}
.cost-combo__control input[type='text']:focus {
  outline: none;
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.15);
}
.cost-combo__clear {
  position: absolute;
  right: 8px;
  top: 50%;
  transform: translateY(-50%);
  display: inline-flex;
  border: none;
  background: none;
  color: var(--admin-muted);
  cursor: pointer;
  padding: 2px;
  border-radius: 6px;
}
.cost-combo__clear:hover { color: var(--admin-text); background: var(--admin-soft); }
.cost-combo__menu {
  position: absolute;
  z-index: 30;
  top: calc(100% + 4px);
  left: 0;
  right: 0;
  max-height: 260px;
  overflow-y: auto;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 12px;
  box-shadow: var(--admin-shadow-soft);
  padding: 4px;
}
.cost-combo__option {
  display: flex;
  flex-direction: column;
  gap: 1px;
  width: 100%;
  text-align: left;
  border: none;
  background: none;
  cursor: pointer;
  padding: 6px 8px;
  border-radius: 8px;
  color: var(--admin-text);
  font-size: 0.85rem;
}
.cost-combo__option:hover,
.cost-combo__option[data-active='true'] { background: var(--admin-soft); }
.cost-combo__option-sub { font-size: 0.72rem; color: var(--admin-muted); }
.cost-combo__empty { padding: 8px; color: var(--admin-muted); font-size: 0.82rem; }

/* ---------------------------------------------------------------------------
   Status badge — мягкие пилюли с тоном.
   --------------------------------------------------------------------------- */

.admin-status-badge {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  padding: 0.2rem 0.65rem;
  border-radius: var(--admin-radius-pill);
  font-size: 0.78rem;
  font-weight: 600;
  letter-spacing: 0.02em;
  background: var(--admin-soft);
  color: var(--admin-muted);
  border: 1px solid transparent;
  white-space: nowrap;
}
.admin-status-badge--success {
  background: var(--admin-success-soft);
  color: var(--admin-success-fg);
}
.admin-status-badge--info {
  background: var(--admin-primary-soft);
  color: var(--admin-primary-fg);
}
.admin-status-badge--warning {
  background: var(--admin-warning-soft);
  color: var(--admin-warning-fg);
}
.admin-status-badge--danger {
  background: var(--admin-danger-soft);
  color: var(--admin-danger-fg);
}
.admin-status-badge--muted {
  background: var(--admin-soft);
  color: var(--admin-muted);
}
.admin-status-badge__dot {
  width: 6px;
  height: 6px;
  border-radius: 50%;
  background: currentColor;
  display: inline-block;
}

/* ---------------------------------------------------------------------------
   Empty state — мягкая «пустая» карточка.
   --------------------------------------------------------------------------- */

.admin-empty {
  background: var(--admin-surface);
  border: 1px dashed var(--admin-border);
  border-radius: var(--admin-radius-card);
  padding: 2.25rem 1.5rem;
  display: grid;
  justify-items: center;
  gap: 0.5rem;
  text-align: center;
}

.admin-empty__icon {
  width: 56px;
  height: 56px;
  border-radius: 50%;
  background: var(--admin-soft);
  color: var(--admin-muted);
  display: inline-flex;
  align-items: center;
  justify-content: center;
}

.admin-empty__title {
  font-weight: 600;
  font-size: 1rem;
  color: var(--admin-text);
}

.admin-empty__hint {
  color: var(--admin-muted);
  font-size: 0.9rem;
  max-width: 40ch;
  line-height: 1.5;
}

.admin-empty__actions {
  margin-top: 0.5rem;
  display: flex;
  gap: 0.5rem;
}

/* ---------------------------------------------------------------------------
   Admin button (light reuse поверх .btn).
   --------------------------------------------------------------------------- */

.admin-btn {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.55rem 1rem;
  border-radius: 12px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font-weight: 500;
  font-size: 0.92rem;
  cursor: pointer;
  text-decoration: none;
  transition: background 0.12s, border-color 0.12s;
}
.admin-btn:hover {
  background: var(--admin-soft);
  border-color: #cbd5e1;
}
.admin-btn--primary {
  background: var(--admin-primary);
  border-color: var(--admin-primary);
  color: #fff;
  box-shadow: 0 6px 18px -8px rgba(37, 99, 235, 0.6);
}
.admin-btn--primary:hover {
  background: #1d4ed8;
  border-color: #1d4ed8;
  color: #fff;
}
.admin-btn--ghost {
  background: transparent;
  border-color: transparent;
  color: var(--admin-primary);
}
.admin-btn--ghost:hover {
  background: var(--admin-primary-soft);
  border-color: transparent;
  color: var(--admin-primary-fg);
}

/*
 * Базовый инпут/селект для inline-форм управленческих карточек
 * (этап «Корректировка материалов после просчёта» и далее). Совпадает
 * по визуалу с `.admin-btn` (рамка/радиус/поверхность), чтобы формы в
 * карточках выглядели единообразно.
 */
.admin-input {
  display: inline-flex;
  width: 100%;
  padding: 0.5rem 0.7rem;
  border-radius: 10px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font-size: 0.85rem;
  line-height: 1.2;
}
.admin-input:focus {
  outline: none;
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px var(--admin-primary-soft);
}
.admin-input:disabled {
  opacity: 0.6;
  cursor: not-allowed;
}

/* ---------------------------------------------------------------------------
   Admin home dashboard — крупные карточки разделов.
   --------------------------------------------------------------------------- */

.admin-home-grid {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(260px, 1fr));
  gap: 1rem;
}

.admin-home-card {
  display: grid;
  gap: 0.75rem;
  padding: 1.25rem 1.4rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: var(--admin-radius-card);
  text-decoration: none;
  color: inherit;
  box-shadow: var(--admin-shadow-soft);
  transition: transform 0.15s ease, box-shadow 0.15s ease,
    border-color 0.15s ease;
}
.admin-home-card:hover {
  transform: translateY(-2px);
  border-color: #cbd5e1;
  box-shadow: 0 12px 28px -16px rgba(15, 23, 42, 0.18);
  color: inherit;
}
.admin-home-card:focus-visible {
  outline: 2px solid var(--admin-primary);
  outline-offset: 2px;
}

.admin-home-card__icon {
  width: 44px;
  height: 44px;
  border-radius: 14px;
  background: var(--admin-primary-soft);
  color: var(--admin-primary-fg);
  display: inline-flex;
  align-items: center;
  justify-content: center;
}

.admin-home-card__title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 600;
  color: var(--admin-text);
}

.admin-home-card__desc {
  margin: 0;
  color: var(--admin-muted);
  font-size: 0.9rem;
  line-height: 1.5;
  min-height: 2.7em;
}

.admin-home-card__cta {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  margin-top: auto;
  color: var(--admin-primary);
  font-weight: 500;
  font-size: 0.9rem;
}

/* ---------------------------------------------------------------------------
   Tech-info collapsible — для «Технической информации» на detail-page.
   --------------------------------------------------------------------------- */

.admin-tech-info {
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  padding: 0.4rem 1rem;
  font-size: 0.88rem;
}
.admin-tech-info > summary {
  cursor: pointer;
  font-weight: 500;
  color: var(--admin-muted);
  padding: 0.6rem 0;
  list-style: none;
}
.admin-tech-info > summary::-webkit-details-marker { display: none; }
.admin-tech-info[open] > summary { color: var(--admin-text); }
.admin-tech-info dl {
  display: grid;
  grid-template-columns: max-content 1fr;
  gap: 0.4rem 1rem;
  margin: 0.5rem 0 0.75rem;
  color: var(--admin-text);
}
.admin-tech-info dt {
  color: var(--admin-muted);
  font-weight: 500;
}
.admin-tech-info dd {
  margin: 0;
  font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, monospace;
  font-size: 0.85rem;
  word-break: break-all;
}

/* ---------------------------------------------------------------------------
   Mobile — таблицы коллапсируются в карточки на узком экране.
   --------------------------------------------------------------------------- */

@media (max-width: 720px) {
  .admin-shell {
    padding: 1rem 0.85rem 2.5rem;
  }
  .admin-table-wrap {
    border-radius: 16px;
  }
  .admin-table thead {
    display: none;
  }
  .admin-table,
  .admin-table tbody,
  .admin-table tr,
  .admin-table td {
    display: block;
    width: 100%;
  }
  .admin-table tr {
    padding: 0.85rem 1rem;
    border-top: 1px solid var(--admin-border);
  }
  .admin-table tbody tr:first-child {
    border-top: 0;
  }
  .admin-table tbody td {
    border: 0;
    padding: 0.2rem 0;
  }
  .admin-table tbody td[data-label]::before {
    content: attr(data-label);
    display: block;
    color: var(--admin-muted);
    font-size: 0.75rem;
    text-transform: uppercase;
    letter-spacing: 0.04em;
    margin-bottom: 0.1rem;
  }
  .admin-table__actions {
    text-align: left;
    margin-top: 0.5rem;
  }
  .admin-home-grid {
    grid-template-columns: 1fr;
  }
}

/* =============================================================================
   Admin UI 2.0 — sidebar layout
   ----------------------------------------------------------------------------
   Двухколоночная раскладка для `/admin/*`: фиксированный sidebar 240 px
   слева, content справа. На ≤ 900 px sidebar скрывается, и сверху
   появляется `<details>`-drawer с теми же ссылками.

   Override-блок ниже снимает с `.app-main` ограничение `max-width: 1280px`
   и убирает горизонтальные паддинги — внутри `/admin` нам нужна вся
   ширина окна, чтобы сайдбар прижимался к левому краю и не «прыгал»
   при ресайзе. `body:has(.admin-layout)` — селектор без JS, поддержан
   во всех современных браузерах (см. caniuse: `:has()` ≥ 2023).
   ============================================================================= */

body:has(.admin-layout) .app-main {
  max-width: none;
  padding: 0;
  background: var(--admin-bg);
}

/* Внутри admin не показываем «расползшийся» admin-блок верхней навигации —
   у него своё место в sidebar. Десктопные ссылки из шапки прячем,
   юзер-блок (имя, выход) оставляем — он нужен на всех страницах. */
body:has(.admin-layout) .app-header__nav { display: none; }

.admin-layout {
  display: grid;
  grid-template-columns: 240px minmax(0, 1fr);
  min-height: calc(100vh - 56px);
  background: var(--admin-bg);
  color: var(--admin-text);
}

.admin-layout__content {
  min-width: 0;
  padding: 0;
}

.admin-layout__mobile-bar {
  display: none;
  padding: 0.65rem 0.85rem;
  background: var(--admin-surface);
  border-bottom: 1px solid var(--admin-border);
  position: sticky;
  top: 56px;
  z-index: 5;
}

/* Sidebar ------------------------------------------------------------------ */

.admin-sidebar {
  position: sticky;
  top: 56px;
  align-self: start;
  height: calc(100vh - 56px);
  background: var(--admin-surface);
  border-right: 1px solid var(--admin-border);
  padding: 1.1rem 0.75rem 1.5rem;
  display: flex;
  flex-direction: column;
  gap: 0.75rem;
  overflow-y: auto;
}

.admin-sidebar__brand {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  padding: 0.4rem 0.6rem 0.7rem;
  margin-bottom: 0.25rem;
  border-bottom: 1px solid var(--admin-border);
  color: var(--admin-text);
  font-weight: 600;
  letter-spacing: -0.01em;
  font-size: 0.95rem;
}

.admin-sidebar__brand-mark {
  width: 28px;
  height: 28px;
  border-radius: 8px;
  background: var(--admin-primary-soft);
  color: var(--admin-primary-fg);
  display: inline-flex;
  align-items: center;
  justify-content: center;
}

.admin-sidebar__nav { flex: 1 1 auto; min-height: 0; overflow-y: auto; }
.admin-sidebar__nav ul {
  list-style: none;
  padding: 0;
  margin: 0;
  display: grid;
  gap: 2px;
}

.admin-sidebar__link {
  display: flex;
  align-items: center;
  gap: 10px;
  padding: 0.55rem 0.7rem;
  border-radius: 10px;
  color: var(--admin-text);
  text-decoration: none;
  font-size: 0.93rem;
  line-height: 1.2;
  transition: background 0.12s ease, color 0.12s ease;
}
.admin-sidebar__link:hover {
  background: var(--admin-soft);
  color: var(--admin-text);
}
.admin-sidebar__link--active,
.admin-sidebar__link--active:hover {
  background: var(--admin-primary-soft);
  color: var(--admin-primary-fg);
  font-weight: 600;
}
.admin-sidebar__link--active .admin-sidebar__icon { color: var(--admin-primary); }

.admin-sidebar__icon {
  width: 22px;
  height: 22px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  color: var(--admin-muted);
  flex: 0 0 auto;
}
.admin-sidebar__link:hover .admin-sidebar__icon { color: var(--admin-text); }

.admin-sidebar__label {
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}

/* Mobile drawer (≤ 900 px) ------------------------------------------------- */

.admin-sidebar-mobile { width: 100%; }

.admin-sidebar-mobile__summary {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.5rem 0.85rem;
  border-radius: 12px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font-weight: 500;
  font-size: 0.95rem;
  cursor: pointer;
  list-style: none;
}
.admin-sidebar-mobile__summary::-webkit-details-marker { display: none; }
.admin-sidebar-mobile__summary:hover { background: var(--admin-soft); }

.admin-sidebar-mobile__panel {
  margin-top: 0.6rem;
  padding: 0.5rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  box-shadow: var(--admin-shadow-soft);
}
.admin-sidebar-mobile__panel ul {
  list-style: none;
  padding: 0;
  margin: 0;
  display: grid;
  gap: 2px;
}

@media (max-width: 900px) {
  .admin-layout {
    grid-template-columns: 1fr;
  }
  .admin-sidebar { display: none; }
  .admin-layout__mobile-bar { display: block; }
}

/* Dashboard stats — счётчик внутри карточки.
   --------------------------------------------------------------------------- */

.admin-home-card__stat {
  display: inline-flex;
  align-items: baseline;
  gap: 0.4rem;
  margin-top: 0.1rem;
  color: var(--admin-text);
  font-weight: 600;
  font-size: 0.95rem;
}
.admin-home-card__stat-num {
  font-size: 1.55rem;
  line-height: 1;
  font-weight: 700;
  letter-spacing: -0.01em;
}
.admin-home-card__stat-unit {
  color: var(--admin-muted);
  font-weight: 500;
  font-size: 0.9rem;
}
.admin-home-card__stat--muted .admin-home-card__stat-num,
.admin-home-card__stat--muted .admin-home-card__stat-unit {
  color: var(--admin-muted);
  font-weight: 500;
}

/* =============================================================================
   Admin UI 2.5 — consistency pass
   ----------------------------------------------------------------------------
   Финальная полировка после Admin UI 2.0:
     - единые токены отступов (`--admin-space-*`);
     - типографика (`.admin-page-title`, `.admin-section-title`, `.admin-muted`);
     - `AdminPageShell` (`.admin-page-shell`) — единая обёртка страниц;
     - формы (`.admin-form`, `.admin-form-grid`, `.admin-field`,
       `.admin-actions-row`);
     - пагинация (`.admin-pagination`);
     - лёгкие анимации (page appear, card hover, button hover).
   На всех `/admin/*` верхний `AppHeader` не рендерится — sidebar
   единственная навигация. Чтобы CSS не оставлял «дыру» сверху, снимаем
   sticky-смещение sidebar/mobile-bar (top: 56 → 0).
   ============================================================================= */

:root {
  --admin-space-xs: 6px;
  --admin-space-sm: 10px;
  --admin-space-md: 16px;
  --admin-space-lg: 24px;
  --admin-space-xl: 32px;
}

/* Header убран на /admin — снимаем sticky-смещение sidebar и mobile-bar,
   которое раньше учитывало 56px шапки. */
body:has(.admin-layout) .app-header { display: none !important; }
.admin-layout {
  min-height: 100vh;
}
.admin-sidebar { top: 0; height: 100vh; }
.admin-layout__mobile-bar { top: 0; }

/* Типографика ----------------------------------------------------------- */

.admin-page-title {
  margin: 0;
  font-size: clamp(1.6rem, 2vw, 2rem);
  line-height: 1.15;
  font-weight: 700;
  letter-spacing: -0.03em;
  color: var(--admin-text);
  display: flex;
  align-items: center;
  gap: var(--admin-space-sm);
}

.admin-section-title {
  margin: 0;
  font-size: 1rem;
  line-height: 1.3;
  font-weight: 650;
  color: var(--admin-text);
}

.admin-muted {
  color: var(--admin-muted);
  line-height: 1.5;
}

/* AdminPageShell -------------------------------------------------------- */

.admin-page-shell {
  padding: var(--admin-space-lg);
  display: grid;
  gap: var(--admin-space-lg);
  color: var(--admin-text);
  animation: admin-page-appear 180ms ease both;
}

.admin-page-shell__header {
  display: flex;
  flex-wrap: wrap;
  gap: var(--admin-space-md);
  justify-content: space-between;
  align-items: flex-start;
}

.admin-page-shell__heading {
  display: flex;
  align-items: center;
  gap: var(--admin-space-sm);
  min-width: 0;
}

.admin-page-shell__icon {
  width: 40px;
  height: 40px;
  border-radius: 12px;
  background: var(--admin-primary-soft);
  color: var(--admin-primary-fg);
  display: inline-flex;
  align-items: center;
  justify-content: center;
  flex: 0 0 auto;
}

.admin-page-shell__title-block { min-width: 0; }

.admin-page-shell__subtitle {
  margin: 4px 0 0;
  color: var(--admin-muted);
  font-size: 0.95rem;
  line-height: 1.5;
  max-width: 90ch;
}

.admin-page-shell__actions {
  display: flex;
  gap: var(--admin-space-sm);
  flex-wrap: wrap;
  align-items: center;
}

@media (max-width: 720px) {
  .admin-page-shell {
    padding: var(--admin-space-md);
    gap: var(--admin-space-md);
  }
  .admin-page-shell__header { flex-direction: column; align-items: stretch; }
  .admin-page-shell__actions { justify-content: flex-start; }
}

/* Layout helpers -------------------------------------------------------- */

.admin-grid-2 {
  display: grid;
  grid-template-columns: minmax(0, 2fr) minmax(0, 1fr);
  gap: var(--admin-space-lg);
  align-items: start;
}

.admin-stack {
  display: grid;
  gap: var(--admin-space-md);
}

@media (max-width: 900px) {
  .admin-grid-2 { grid-template-columns: 1fr; }
}

/* Card hover ------------------------------------------------------------ */

.admin-card {
  transition: border-color 160ms ease, box-shadow 160ms ease,
    transform 160ms ease;
}
.admin-card:hover {
  border-color: #cfd8e3;
  box-shadow: 0 4px 16px rgba(15, 23, 42, 0.05);
}

/* Buttons --------------------------------------------------------------- */

.admin-btn { transition: background 160ms ease, border-color 160ms ease,
  color 160ms ease, transform 120ms ease, box-shadow 160ms ease; }
.admin-btn:hover { transform: translateY(-1px); }
.admin-btn:active { transform: translateY(0); }
.admin-btn--danger {
  background: var(--admin-danger);
  border-color: var(--admin-danger);
  color: #fff;
}
.admin-btn--danger:hover {
  background: #b91c1c;
  border-color: #b91c1c;
  color: #fff;
}

/* Sidebar item hover (мягкий fade) ------------------------------------- */

.admin-sidebar__link { transition: background 160ms ease, color 160ms ease; }

/* Forms ----------------------------------------------------------------- */

.admin-form {
  display: grid;
  gap: var(--admin-space-md);
}

.admin-form-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(220px, 1fr));
  gap: var(--admin-space-md);
}

/* -----------------------------------------------------------------------------
   Сетка формы принтера (`/admin/printers/new`, `/admin/printers/[id]`).
   Три поля «Название | Тип | Роль» должны стоять в одну линию на широких
   экранах (так короче и читабельнее), но на планшете/мобильном — переходить
   в столбец, иначе селекты сжимаются до неюзабельных 90–110px.

   Градации:
     • >= 720px  →  2fr 1fr 2fr   (Название шире, Тип компактный, Роль шире)
     • < 720px   →  1fr           (одна колонка, поля одно под другим)

   `minmax(0, …)` нужен, чтобы длинные опции <option> не растягивали
   колонку и не «прыгал» layout при выборе значения.
   ============================================================================= */
.admin-form-grid--printer-row {
  grid-template-columns: minmax(0, 2fr) minmax(0, 1fr) minmax(0, 2fr);
  align-items: start;
}

@media (max-width: 720px) {
  .admin-form-grid--printer-row {
    grid-template-columns: 1fr;
  }
}

.admin-field {
  display: grid;
  gap: var(--admin-space-xs);
}

.admin-field > label {
  font-size: 0.85rem;
  font-weight: 600;
  color: var(--admin-text);
  letter-spacing: 0.01em;
}

.admin-field > input[type='text'],
.admin-field > input[type='number'],
.admin-field > input[type='password'],
.admin-field > input[type='email'],
.admin-field > input[type='date'],
.admin-field > select,
.admin-field > textarea {
  padding: 0.6rem 0.75rem;
  border-radius: 10px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font: inherit;
  width: 100%;
  transition: border-color 160ms ease, box-shadow 160ms ease;
}

.admin-field > input:focus,
.admin-field > select:focus,
.admin-field > textarea:focus {
  outline: none;
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.15);
}

.admin-field--inline {
  display: flex;
  align-items: center;
  gap: 0.5rem;
}

/* Комбобокс выбора техкарты с группировкой по группам номенклатуры
   (см. `app/admin/orders/new/tech-card-combobox.tsx`). Папки-группы
   разворачиваются по клику, сверху — строка поиска. Внешне совпадает
   с обычным `.admin-field > select` в закрытом состоянии. */
.tc-combobox {
  position: relative;
  width: 100%;
}

.tc-combobox__trigger {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  width: 100%;
  padding: 0.6rem 0.75rem;
  border-radius: 10px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font: inherit;
  text-align: left;
  cursor: pointer;
  transition: border-color 160ms ease, box-shadow 160ms ease;
}

.tc-combobox__trigger:focus-visible {
  outline: none;
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.15);
}

.tc-combobox__value {
  flex: 1 1 auto;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}

.tc-combobox__value--placeholder {
  color: var(--admin-muted);
}

.tc-combobox__caret {
  flex: 0 0 auto;
  color: var(--admin-muted);
}

.tc-combobox__panel {
  position: absolute;
  z-index: 30;
  top: calc(100% + 4px);
  left: 0;
  right: 0;
  display: flex;
  flex-direction: column;
  max-height: 320px;
  border-radius: 10px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  box-shadow: 0 12px 32px rgba(15, 23, 42, 0.18);
  overflow: hidden;
}

.tc-combobox__search {
  display: flex;
  align-items: center;
  gap: 0.45rem;
  padding: 0.5rem 0.6rem;
  border-bottom: 1px solid var(--admin-border);
  color: var(--admin-muted);
}

.tc-combobox__search-input {
  flex: 1 1 auto;
  border: none;
  background: transparent;
  color: var(--admin-text);
  font: inherit;
  outline: none;
}

.tc-combobox__search-clear {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  border: none;
  background: transparent;
  color: var(--admin-muted);
  cursor: pointer;
  padding: 0.1rem;
  border-radius: 6px;
}

.tc-combobox__search-clear:hover {
  color: var(--admin-text);
  background: rgba(148, 163, 184, 0.18);
}

.tc-combobox__list {
  overflow-y: auto;
  padding: 0.25rem;
}

.tc-combobox__empty {
  padding: 0.75rem;
  color: var(--admin-muted);
  font-size: 0.88rem;
  text-align: center;
}

.tc-combobox__folder {
  display: flex;
  align-items: center;
  gap: 0.45rem;
  width: 100%;
  padding: 0.45rem 0.5rem;
  border: none;
  background: transparent;
  color: var(--admin-text);
  font: inherit;
  font-weight: 600;
  text-align: left;
  cursor: pointer;
  border-radius: 8px;
}

.tc-combobox__folder:hover {
  background: rgba(148, 163, 184, 0.14);
}

.tc-combobox__folder-icon {
  flex: 0 0 auto;
  color: var(--admin-primary);
}

.tc-combobox__folder-name {
  flex: 1 1 auto;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}

.tc-combobox__folder-count {
  flex: 0 0 auto;
  font-size: 0.78rem;
  font-weight: 600;
  color: var(--admin-muted);
  background: rgba(148, 163, 184, 0.16);
  border-radius: 999px;
  padding: 0.05rem 0.45rem;
}

.tc-combobox__cards {
  display: flex;
  flex-direction: column;
  padding-left: 1.35rem;
}

.tc-combobox__card {
  display: block;
  width: 100%;
  padding: 0.4rem 0.5rem;
  border: none;
  background: transparent;
  color: var(--admin-text);
  font: inherit;
  text-align: left;
  cursor: pointer;
  border-radius: 8px;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}

.tc-combobox__card:hover {
  background: rgba(148, 163, 184, 0.14);
}

.tc-combobox__card.is-selected {
  background: rgba(37, 99, 235, 0.12);
  color: var(--admin-primary);
  font-weight: 600;
}

.tc-combobox__card--reset {
  color: var(--admin-muted);
}

.admin-field__hint {
  color: var(--admin-muted);
  font-size: 0.82rem;
  line-height: 1.45;
}

.admin-actions-row {
  display: flex;
  gap: var(--admin-space-sm);
  justify-content: flex-end;
  flex-wrap: wrap;
  align-items: center;
}

.admin-actions-row--split {
  justify-content: space-between;
}

.admin-actions-row__danger {
  margin-right: auto;
}

/* Tech-card material requirements row (UI 2.7 «Техкарта =
   материальные требования»):
   - блок переименован в «Материальные требования»;
   - legacy поля строки (`name` / `unit` / `qtyPerUnit` / `note`)
     скрыты, видимыми остаются только: роль материала, характеристика
     полотна, плотность, ширина рулона, правило цвета, фиксированный
     цвет;
   - сетка адаптивна: 2–3 колонки на средних экранах, одна — на
     мобильных, без громоздких вертикальных отступов под legacy. */

.admin-material-requirements__hint {
  max-width: 70ch;
}

.admin-material-row {
  /* Slightly tighter than default `.admin-card`, чтобы строки
     материальных требований не «дышали» как полноценные карточки. */
  border-color: var(--admin-border);
}

.admin-material-row__grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
  gap: var(--admin-space-sm);
  align-items: end;
}

.admin-material-row__remove {
  align-self: end;
  justify-self: end;
}

@media (max-width: 700px) {
  .admin-material-row__grid {
    grid-template-columns: 1fr;
  }
  .admin-material-row__remove {
    justify-self: start;
  }
}

.admin-tech-card-outsource-legacy__hint {
  max-width: 70ch;
}

/* QR preview ------------------------------------------------------------ */

.admin-qr-card { display: grid; gap: var(--admin-space-md); }
.admin-qr-card__body {
  display: flex;
  align-items: center;
  gap: var(--admin-space-md);
  flex-wrap: wrap;
}
.admin-qr-card__image {
  width: 160px;
  height: 160px;
  border: 1px solid var(--admin-border);
  border-radius: 12px;
  background: #fff;
  image-rendering: pixelated;
  flex: 0 0 auto;
}
.admin-qr-card__actions {
  display: flex;
  gap: var(--admin-space-sm);
  flex-wrap: wrap;
}

/* Definition list (кратко) --------------------------------------------- */

.admin-deflist {
  display: grid;
  grid-template-columns: max-content 1fr;
  gap: 0.5rem var(--admin-space-md);
  margin: 0;
}
.admin-deflist dt {
  color: var(--admin-muted);
  font-size: 0.85rem;
  font-weight: 500;
  align-self: center;
}
.admin-deflist dd {
  margin: 0;
  color: var(--admin-text);
  font-size: 0.95rem;
  font-weight: 500;
  word-break: break-word;
}

/* Pagination ------------------------------------------------------------ */

.admin-pagination {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: var(--admin-space-md);
  flex-wrap: wrap;
  padding: var(--admin-space-sm) 0 0;
  color: var(--admin-muted);
  font-size: 0.88rem;
}

.admin-pagination__range { color: var(--admin-muted); }

.admin-pagination__nav {
  display: inline-flex;
  align-items: center;
  gap: var(--admin-space-xs);
}

.admin-pagination__btn {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  padding: 0.4rem 0.7rem;
  border-radius: 8px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font-size: 0.88rem;
  text-decoration: none;
  cursor: pointer;
  transition: background 160ms ease, border-color 160ms ease, color 160ms ease;
}
.admin-pagination__btn:hover {
  background: var(--admin-soft);
  border-color: #cbd5e1;
}
.admin-pagination__btn[aria-disabled='true'],
.admin-pagination__btn:disabled {
  opacity: 0.45;
  pointer-events: none;
}

.admin-pagination__size {
  display: inline-flex;
  align-items: center;
  gap: var(--admin-space-xs);
}

.admin-pagination__size select {
  padding: 0.35rem 0.5rem;
  border-radius: 8px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font: inherit;
  font-size: 0.85rem;
}

/* Tabs ------------------------------------------------------------------ */

.admin-tabs {
  display: inline-flex;
  gap: var(--admin-space-xs);
  margin-bottom: var(--admin-space-md);
  flex-wrap: wrap;
}

.admin-tab {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.4rem 0.85rem;
  border-radius: 8px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-muted);
  font-size: 0.88rem;
  text-decoration: none;
  transition: background 160ms ease, color 160ms ease, border-color 160ms ease;
}

.admin-tab:hover {
  color: var(--admin-text);
  border-color: var(--admin-border-strong, var(--admin-border));
}

.admin-tab--active {
  background: var(--admin-primary-soft);
  color: var(--admin-text);
  border-color: var(--admin-primary, var(--admin-border));
}

/* Animations ------------------------------------------------------------ */

/*
 * Сознательно opacity-only, без `transform`. Любое значение
 * `transform` (даже translateY(0)) превращает элемент в containing
 * block для `position: fixed` потомков (см. CSS Containing Block §4),
 * а `animation: ... both` приклеивает финальный кадр навсегда.
 * В этом проекте такая ловушка уже один раз ломала overlay-модалки
 * внутри `.admin-page-shell` (модалка bulk-print «Печать ячеек» уезжала
 * к центру длинной страницы вместо центра viewport — см.
 * `docs/warehouse-bulk-print-modal-runtime-recon.md`). Не возвращать
 * `transform` сюда, иначе ловушка вернётся для будущих fixed-потомков.
 */
@keyframes admin-page-appear {
  from { opacity: 0; }
  to { opacity: 1; }
}

@media (prefers-reduced-motion: reduce) {
  .admin-page-shell,
  .admin-card,
  .admin-btn,
  .admin-sidebar__link,
  .admin-home-card,
  .admin-pagination__btn {
    animation: none !important;
    transition: none !important;
  }
}

/* =============================================================================
   Admin Analytics — KPI + Heatmap + Creative color system
   ----------------------------------------------------------------------------
   Расширение Admin UI 2.5: цветовые акценты по семантике (заказы синие,
   выпуск зелёный, WIP оранжевый, диагностика/мастер фиолетовый), KPI-
   карточки на /admin, тепловая карта потока операций.

   Принципы:
     - дизайн светлый, soft pastel — никаких heavy backgrounds;
     - акценты через soft-fill + чёткий иконочный bubble;
     - переходы 160ms ease, prefers-reduced-motion уважаем (см. ниже).
   ============================================================================= */

:root {
  --admin-blue: #2563eb;
  --admin-blue-soft: #dbeafe;
  --admin-blue-fg: #1e3a8a;

  --admin-green: #16a34a;
  --admin-green-soft: #dcfce7;
  --admin-green-fg: #166534;

  --admin-orange: #f97316;
  --admin-orange-soft: #ffedd5;
  --admin-orange-fg: #9a3412;

  --admin-purple: #7c3aed;
  --admin-purple-soft: #ede9fe;
  --admin-purple-fg: #5b21b6;

  /** «Узкое место» — отдельный коралловый highlight, чтобы он отличался
      от обычных оранжевых WIP-чипов и не сливался с ними на heatmap. */
  --admin-coral: #fb7185;
  --admin-coral-soft: #ffe4e6;
  --admin-coral-fg: #9f1239;
}

/* Универсальные accent-классы — bubble icon + soft fg.
   Используются на dashboard cards и любых вспомогательных бейджах.
   --------------------------------------------------------------------------- */

.admin-accent-blue {
  background: var(--admin-blue-soft);
  color: var(--admin-blue-fg);
}
.admin-accent-green {
  background: var(--admin-green-soft);
  color: var(--admin-green-fg);
}
.admin-accent-orange {
  background: var(--admin-orange-soft);
  color: var(--admin-orange-fg);
}
.admin-accent-purple {
  background: var(--admin-purple-soft);
  color: var(--admin-purple-fg);
}
.admin-accent-coral {
  background: var(--admin-coral-soft);
  color: var(--admin-coral-fg);
}

/* KPI-карточки (`/admin` блок «Сегодня в производстве»).
   --------------------------------------------------------------------------- */

.admin-kpi-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
  gap: var(--admin-space-md);
}

.admin-kpi-card {
  position: relative;
  display: grid;
  gap: 6px;
  padding: var(--admin-space-md) var(--admin-space-md);
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 16px;
  box-shadow: var(--admin-shadow-soft);
  overflow: hidden;
  animation: admin-kpi-appear 220ms ease both;
  transition: transform 160ms ease, box-shadow 160ms ease,
    border-color 160ms ease;
}

.admin-kpi-card::before {
  content: '';
  position: absolute;
  inset: 0 auto 0 0;
  width: 4px;
  background: var(--admin-border);
  border-radius: 4px 0 0 4px;
}

.admin-kpi-card--blue::before { background: var(--admin-blue); }
.admin-kpi-card--green::before { background: var(--admin-green); }
.admin-kpi-card--orange::before { background: var(--admin-orange); }
.admin-kpi-card--purple::before { background: var(--admin-purple); }
.admin-kpi-card--coral::before { background: var(--admin-coral); }

.admin-kpi-card:hover {
  transform: translateY(-1px);
  border-color: #cfd8e3;
  box-shadow: 0 8px 24px -16px rgba(15, 23, 42, 0.18);
}

.admin-kpi-card__head {
  display: flex;
  align-items: center;
  gap: var(--admin-space-sm);
  color: var(--admin-muted);
  font-size: 0.82rem;
  font-weight: 600;
  letter-spacing: 0.02em;
  text-transform: uppercase;
}

.admin-kpi-card__bubble {
  width: 28px;
  height: 28px;
  border-radius: 8px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  flex: 0 0 auto;
}

.admin-kpi-card__value {
  font-size: 1.7rem;
  line-height: 1.05;
  font-weight: 700;
  letter-spacing: -0.02em;
  color: var(--admin-text);
}

.admin-kpi-card__sub {
  color: var(--admin-muted);
  font-size: 0.82rem;
  line-height: 1.35;
}

/* Heatmap — горизонтальные чипы операций маршрута.
   --------------------------------------------------------------------------- */

.admin-heatmap {
  display: grid;
  gap: var(--admin-space-md);
}

.admin-heatmap__row {
  display: flex;
  flex-wrap: wrap;
  gap: var(--admin-space-sm);
}

.admin-heatmap__chip {
  display: inline-grid;
  grid-template-rows: auto auto;
  gap: 2px;
  align-items: start;
  min-width: 132px;
  padding: 10px 12px;
  border-radius: 12px;
  border: 1px solid transparent;
  background: var(--admin-soft);
  color: var(--admin-muted);
  text-decoration: none;
  font-size: 0.85rem;
  line-height: 1.2;
  transition: transform 160ms ease, box-shadow 160ms ease,
    border-color 160ms ease;
}

.admin-heatmap__chip:hover {
  transform: translateY(-1px);
  box-shadow: 0 4px 12px rgba(15, 23, 42, 0.06);
}

.admin-heatmap__chip-label {
  font-weight: 600;
  color: inherit;
  letter-spacing: 0.01em;
  white-space: nowrap;
}

.admin-heatmap__chip-stats {
  display: inline-flex;
  align-items: center;
  gap: 8px;
  font-size: 0.8rem;
  font-weight: 500;
  color: inherit;
  opacity: 0.95;
}

.admin-heatmap__chip-stat {
  display: inline-flex;
  align-items: center;
  gap: 3px;
}

.admin-heatmap__chip-stat strong {
  font-weight: 700;
}

/* Цветовые градации:
     - muted — total = 0;
     - blue  — 1..9;
     - green — 10..29;
     - orange — 30+;
     - coral — bottleneck (узкое место). */

.admin-heatmap__chip--muted {
  background: var(--admin-soft);
  color: var(--admin-muted);
  border-color: var(--admin-border);
}
.admin-heatmap__chip--blue {
  background: var(--admin-blue-soft);
  color: var(--admin-blue-fg);
  border-color: rgba(37, 99, 235, 0.18);
}
.admin-heatmap__chip--green {
  background: var(--admin-green-soft);
  color: var(--admin-green-fg);
  border-color: rgba(22, 163, 74, 0.18);
}
.admin-heatmap__chip--orange {
  background: var(--admin-orange-soft);
  color: var(--admin-orange-fg);
  border-color: rgba(249, 115, 22, 0.22);
}
.admin-heatmap__chip--coral {
  background: var(--admin-coral-soft);
  color: var(--admin-coral-fg);
  border-color: rgba(251, 113, 133, 0.32);
  box-shadow: inset 0 0 0 1px rgba(251, 113, 133, 0.18);
}

.admin-heatmap__legend {
  display: inline-flex;
  flex-wrap: wrap;
  gap: var(--admin-space-sm);
  margin-top: 4px;
  color: var(--admin-muted);
  font-size: 0.78rem;
}

.admin-heatmap__legend-dot {
  display: inline-flex;
  align-items: center;
  gap: 4px;
}

.admin-heatmap__legend-swatch {
  display: inline-block;
  width: 10px;
  height: 10px;
  border-radius: 3px;
  border: 1px solid rgba(15, 23, 42, 0.05);
}

/* Dashboard cards redesign — soft-bubble icon с разноцветными акцентами.
   Не ломаем существующие стили `.admin-home-card`, только добавляем
   модификаторы для bubble + meta-row.
   --------------------------------------------------------------------------- */

.admin-home-card__icon--blue {
  background: var(--admin-blue-soft);
  color: var(--admin-blue-fg);
}
.admin-home-card__icon--green {
  background: var(--admin-green-soft);
  color: var(--admin-green-fg);
}
.admin-home-card__icon--orange {
  background: var(--admin-orange-soft);
  color: var(--admin-orange-fg);
}
.admin-home-card__icon--purple {
  background: var(--admin-purple-soft);
  color: var(--admin-purple-fg);
}

.admin-home-card__cta {
  transition: transform 160ms ease, gap 160ms ease;
}
.admin-home-card:hover .admin-home-card__cta {
  gap: 0.5rem;
}

@keyframes admin-kpi-appear {
  from { opacity: 0; transform: translateY(4px); }
  to { opacity: 1; transform: translateY(0); }
}

@media (prefers-reduced-motion: reduce) {
  .admin-kpi-card,
  .admin-heatmap__chip {
    animation: none !important;
    transition: none !important;
  }
}

/* Mobile heatmap — горизонтальный скролл, чтобы чипы не скукоживались. */
@media (max-width: 720px) {
  .admin-heatmap__row {
    flex-wrap: nowrap;
    overflow-x: auto;
    padding-bottom: 4px;
    -webkit-overflow-scrolling: touch;
  }
  .admin-heatmap__chip {
    flex: 0 0 auto;
  }
}

/* ===========================================================================
   Admin Final Cleanup (Admin UI 2.6)

   Компактные «чипы», цепочка шагов маршрута и нижний блок sidebar
   с logout. Никакой dark-темы / агрессивных hover’ов — soft surfaces,
   border-radius 999px, font-size 0.78–0.95rem. Все классы документированы
   в `docs/admin-ui-2.6.md`.
   =========================================================================== */

/* --- Sidebar footer (Logout) -------------------------------------------- */

.admin-sidebar__footer {
  border-top: 1px solid var(--admin-border);
  padding-top: 0.6rem;
  margin-top: 0.4rem;
  flex: 0 0 auto;
  /* Stack pinned link (Настройки) и кнопку «Выйти» с тем же gap-ом,
     что у nav-ссылок выше — чтобы футер визуально продолжал список. */
  display: flex;
  flex-direction: column;
  gap: 2px;
}
.admin-sidebar-mobile__footer {
  border-top: 1px solid var(--admin-border);
  margin-top: 0.5rem;
  padding-top: 0.5rem;
  display: flex;
  flex-direction: column;
  gap: 2px;
}
.admin-sidebar__logout-form { width: 100%; }
.admin-sidebar__logout {
  display: flex;
  width: 100%;
  align-items: center;
  gap: 10px;
  padding: 0.55rem 0.7rem;
  border-radius: 10px;
  border: 0;
  background: transparent;
  color: var(--admin-muted);
  font-size: 0.93rem;
  line-height: 1.2;
  cursor: pointer;
  text-align: left;
  transition: background 0.12s ease, color 0.12s ease;
}
.admin-sidebar__logout:hover {
  background: var(--admin-soft);
  color: var(--admin-text);
}
.admin-sidebar__logout:focus-visible {
  outline: 2px solid var(--admin-primary);
  outline-offset: 2px;
}

/* --- Compact chip lists -------------------------------------------------- */

.admin-chip-list {
  display: flex;
  flex-wrap: wrap;
  gap: 6px;
  list-style: none;
  padding: 0;
  margin: 0;
}
.admin-chip {
  display: inline-flex;
  align-items: center;
  gap: 6px;
  height: 30px;
  padding: 0 10px;
  border-radius: 999px;
  background: var(--admin-soft);
  color: var(--admin-text);
  font-size: 0.85rem;
  line-height: 1;
  border: 1px solid var(--admin-border);
}
.admin-chip__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  color: var(--admin-muted);
}
.admin-chip__remove {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 18px;
  height: 18px;
  border: 0;
  border-radius: 999px;
  background: transparent;
  color: var(--admin-muted);
  cursor: pointer;
  padding: 0;
  margin-left: 2px;
}
.admin-chip__remove:hover {
  background: var(--admin-border);
  color: var(--admin-text);
}
.admin-chip-add {
  display: inline-flex;
  align-items: center;
  gap: 8px;
  flex-wrap: wrap;
}
.admin-chip-add__select {
  height: 32px;
  padding: 0 0.6rem;
  border-radius: 999px;
  border: 1px solid var(--admin-border);
  background: var(--admin-surface);
  color: var(--admin-text);
  font-size: 0.88rem;
  min-width: 220px;
}

/* --- Compact grouped table (admin-operations / admin-equipment) ---------
   ОДНА карточка + ОДИН table header на всю страницу, категории живут
   как тонкие group-row внутри tbody (см. ТЗ «compact grouped-table
   layout»). Раньше каждая категория была отдельным `CategorySection`
   с собственным заголовком/таблицей, и страницы /admin/operations
   и /admin/equipment имели много вертикального воздуха. Здесь
   уплотняем padding, убираем повторяющиеся header'ы, оставляем
   group-row только как разделитель.

   `.admin-compact-grouped-table` наследует `.admin-table` (border,
   радиусы), но переопределяет padding/font-size — поэтому декларации
   ниже идут после базовых правил `.admin-table` в файле.
   ---------------------------------------------------------------------- */

.admin-compact-grouped-card {
  padding: 0;
  overflow: hidden;
}

.admin-compact-table-wrap {
  overflow-x: auto;
}

.admin-compact-grouped-table {
  margin: 0;
  width: 100%;
}

.admin-compact-grouped-table thead th {
  padding: 0.55rem 0.75rem;
  font-size: 0.72rem;
  white-space: nowrap;
}

.admin-compact-grouped-table tbody td {
  padding: 0.55rem 0.75rem;
  vertical-align: middle;
}

/* Group-row: тонкий разделитель с лейблом категории и счётчиком.
   Подсветку hover-строки внутри `.admin-table` гасим — этот ряд не
   кликабелен и не должен вести себя как обычная строка таблицы. */
.admin-compact-group-row td {
  padding: 0;
  background: var(--admin-soft, #f8fafc);
  border-top: 1px solid var(--admin-border);
  border-bottom: 1px solid var(--admin-border);
}
.admin-compact-grouped-table tbody tr.admin-compact-group-row:hover td {
  background: var(--admin-soft, #f8fafc);
}
.admin-compact-grouped-table tbody tr.admin-compact-group-row:first-child td {
  border-top: 0;
}

.admin-compact-group-row__inner {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.75rem;
  padding: 0.5rem 0.75rem;
  font-weight: 700;
  color: var(--admin-text);
}

.admin-compact-group-row__count {
  font-size: 0.8rem;
  color: var(--admin-muted);
  font-weight: 600;
}

.admin-compact-row .admin-status-badge {
  white-space: nowrap;
}

.admin-compact-row__actions {
  text-align: right;
  white-space: nowrap;
}

/* Категории оборудования — компактные chips в табличной ячейке. */
.admin-equipment-category-chips {
  display: flex;
  flex-wrap: wrap;
  gap: 0.25rem;
  list-style: none;
  margin: 0;
  padding: 0;
}

.admin-equipment-category-chip {
  display: inline-flex;
  align-items: center;
  border-radius: 999px;
  padding: 0.15rem 0.45rem;
  font-size: 0.72rem;
  font-weight: 600;
  background: #eef2ff;
  color: #3730a3;
}

.admin-equipment-display-number {
  font-size: 1.05rem;
  white-space: nowrap;
}

@media (max-width: 720px) {
  /* На узком экране обычная `.admin-table` коллапсируется в блочную
     раскладку (см. media-query выше) — это ломает colspan group-row.
     Принудительно возвращаем table-row/table-cell для compact таблицы
     и оставляем horizontal scroll вместо «карточек». */
  .admin-compact-grouped-table,
  .admin-compact-grouped-table thead,
  .admin-compact-grouped-table tbody,
  .admin-compact-grouped-table tr,
  .admin-compact-grouped-table th,
  .admin-compact-grouped-table td {
    display: revert;
  }
  .admin-compact-grouped-table thead {
    display: table-header-group;
  }
  .admin-compact-grouped-table tbody {
    display: table-row-group;
  }
  .admin-compact-grouped-table tr {
    display: table-row;
    padding: 0;
    border-top: 0;
  }
  .admin-compact-grouped-table th {
    display: table-cell;
  }
  .admin-compact-grouped-table td {
    display: table-cell;
  }
  .admin-compact-grouped-table td[data-label]::before {
    content: none;
  }
  .admin-operations-compact-table {
    min-width: 760px;
  }
  .admin-equipment-compact-table {
    min-width: 860px;
  }
  .admin-compact-group-row td {
    padding: 0;
  }
}

/* --- Compact list / metric / field rows --------------------------------- */

.admin-compact-list {
  display: grid;
  gap: 8px;
  margin: 0;
  padding: 0;
  list-style: none;
}
.admin-metric-row {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 12px;
  padding: 6px 0;
  border-bottom: 1px solid var(--admin-border);
}
.admin-metric-row:last-child { border-bottom: 0; }
.admin-metric-row__label {
  color: var(--admin-muted);
  font-size: 0.78rem;
  text-transform: uppercase;
  letter-spacing: 0.04em;
}
.admin-metric-row__value {
  color: var(--admin-text);
  font-size: 0.95rem;
  font-weight: 500;
}
.admin-field-list {
  display: grid;
  gap: 10px;
  margin: 0;
  padding: 0;
  list-style: none;
}
.admin-field-row {
  display: grid;
  grid-template-columns: minmax(120px, 0.4fr) 1fr;
  gap: 10px;
  align-items: baseline;
}
.admin-field-row__label {
  color: var(--admin-muted);
  font-size: 0.78rem;
}
.admin-field-row__value {
  color: var(--admin-text);
  font-size: 0.95rem;
}

/* --- Route step chain --------------------------------------------------- */

.admin-route-steps {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 6px 4px;
}
.admin-route-steps__item {
  display: inline-flex;
  align-items: center;
  gap: 4px;
}
.admin-route-steps__sep {
  color: var(--admin-muted);
  font-size: 0.85rem;
  padding: 0 2px;
  -webkit-user-select: none;
          user-select: none;
}
/* ⇄ между взаимозаменяемыми операциями (порядок любой). */
.admin-route-steps__sep--parallel {
  display: inline-flex;
  align-items: center;
  color: var(--admin-primary);
}
.admin-route-step {
  display: inline-flex;
  align-items: center;
  gap: 6px;
  height: 28px;
  padding: 0 10px 0 6px;
  border-radius: 999px;
  background: var(--admin-soft);
  color: var(--admin-text);
  font-size: 0.85rem;
  line-height: 1;
  border: 1px solid var(--admin-border);
  white-space: nowrap;
}
.admin-route-step__num {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  min-width: 18px;
  height: 18px;
  padding: 0 5px;
  border-radius: 999px;
  background: var(--admin-surface);
  color: var(--admin-muted);
  font-size: 0.72rem;
  font-weight: 600;
  border: 1px solid var(--admin-border);
}
.admin-route-step__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  color: var(--admin-muted);
}
.admin-route-step__name {
  color: var(--admin-text);
  font-weight: 500;
}
.admin-route-step--more {
  color: var(--admin-muted);
  font-style: italic;
}
.admin-route-step--blue {
  background: var(--admin-blue-soft);
  color: var(--admin-blue-fg);
  border-color: transparent;
}
.admin-route-step--blue .admin-route-step__icon { color: var(--admin-blue-fg); }
.admin-route-step--blue .admin-route-step__name { color: var(--admin-blue-fg); }
.admin-route-step--green {
  background: var(--admin-green-soft);
  color: var(--admin-green-fg);
  border-color: transparent;
}
.admin-route-step--green .admin-route-step__icon { color: var(--admin-green-fg); }
.admin-route-step--green .admin-route-step__name { color: var(--admin-green-fg); }
.admin-route-step--orange {
  background: var(--admin-orange-soft);
  color: var(--admin-orange-fg);
  border-color: transparent;
}
.admin-route-step--orange .admin-route-step__icon { color: var(--admin-orange-fg); }
.admin-route-step--orange .admin-route-step__name { color: var(--admin-orange-fg); }
.admin-route-step--purple {
  background: var(--admin-purple-soft);
  color: var(--admin-purple-fg);
  border-color: transparent;
}
.admin-route-step--purple .admin-route-step__icon { color: var(--admin-purple-fg); }
.admin-route-step--purple .admin-route-step__name { color: var(--admin-purple-fg); }
.admin-route-step--coral {
  background: #fde2e2;
  color: #9b1c1c;
  border-color: transparent;
}
.admin-route-step--coral .admin-route-step__icon { color: #9b1c1c; }
.admin-route-step--coral .admin-route-step__name { color: #9b1c1c; }

.admin-route-steps--dense .admin-route-step {
  height: 24px;
  font-size: 0.78rem;
  padding: 0 8px 0 4px;
}
.admin-route-steps--dense .admin-route-step__num {
  min-width: 16px;
  height: 16px;
  font-size: 0.66rem;
}
.admin-route-steps--dense .admin-route-steps__sep { font-size: 0.78rem; }

/* --- Minor compact form helpers ------------------------------------------ */

.admin-step-edit {
  display: grid;
  gap: 6px;
}
.admin-step-edit__add {
  display: inline-flex;
  align-items: center;
  gap: 8px;
  flex-wrap: wrap;
}

/* Параллельный (взаимозаменяемый) шаг: связан с шагом выше в один этап.
   Лёгкий левый акцент + чип «↕ параллельно» — порядок внутри пары любой. */
.admin-step-edit__row--parallel {
  border-left: 3px solid var(--admin-primary);
  padding-left: 6px;
  border-radius: 4px;
  background: rgba(37, 99, 235, 0.06);
}
.admin-step-edit__parallel-tag {
  display: inline-flex;
  align-items: center;
  gap: 3px;
  margin-left: 8px;
  font-size: 0.72rem;
  font-weight: 600;
  color: var(--admin-primary);
  white-space: nowrap;
}
.admin-btn--active {
  color: var(--admin-primary);
  background: rgba(37, 99, 235, 0.12);
}

/* Admin date field --------------------------------------------------------
 *
 * AdminDateField — единый «красивый» date picker (см.
 * `apps/web/components/admin/admin-date-field.tsx`). Снаружи
 * это `<input type="date">` + кнопка-иконка `Calendar`, которая
 * открывает native picker через `showPicker()`. Стилизация
 * совпадает с прочими `.admin-field > input`, но радиус
 * чуть крупнее (14px) — чтобы дата выделялась как «крупный»
 * управляющий элемент в карточке заказа.
 */

.admin-date-field {
  position: relative;
  display: inline-flex;
  align-items: stretch;
  width: 100%;
  min-height: 40px;
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  background: var(--admin-surface);
  transition: border-color 160ms ease, box-shadow 160ms ease;
  overflow: hidden;
}

.admin-date-field:focus-within {
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.15);
}

.admin-date-field[data-disabled='true'] {
  opacity: 0.6;
  cursor: not-allowed;
}

.admin-date-field__input {
  flex: 1 1;
  border: none;
  outline: none;
  background: transparent;
  color: var(--admin-text);
  font: inherit;
  padding: 0.55rem 0.75rem;
  width: 100%;
  /* Скрываем нативную иконку календаря у Chromium/Edge — мы рисуем
     свою справа, и две иконки подряд выглядят шумно. */
  appearance: none;
}

.admin-date-field__input::-webkit-calendar-picker-indicator {
  opacity: 0;
  width: 0;
  height: 0;
  pointer-events: none;
}

.admin-date-field__button {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 38px;
  border: none;
  border-left: 1px solid var(--admin-border);
  background: transparent;
  color: var(--admin-muted);
  cursor: pointer;
  transition: background 160ms ease, color 160ms ease;
}

.admin-date-field__button:hover:not(:disabled) {
  background: rgba(37, 99, 235, 0.08);
  color: var(--admin-primary);
}

.admin-date-field__button:disabled {
  cursor: not-allowed;
  color: var(--admin-muted);
}

/* Когда AdminDateField лежит внутри `.admin-field`, родительский селектор
   `> input[type='date']` мог бы накинуть конфликтующие стили — но input
   завёрнут в `<span class="admin-date-field">`, так что
   `> input` не сработает. Оставляем явное правило на случай миграции. */
.admin-field > .admin-date-field {
  width: 100%;
}

/* =============================================================================
   Order Deadlines + Risk Detection
   --------------------------------------------------------------------------
   Управленческий слой контроля сроков (см. `@sewing/shared/order-deadlines`).
   Здесь живут все визуальные классы, которые рендерят:
     - `/admin` — KPI-полоса «Контроль сроков» (`.admin-deadline-kpis*`);
     - `/admin/orders` — табы фильтра (`.admin-deadline-tabs*`) и колонка
       «Срок» в таблице (`.admin-deadline-cell*`);
     - `/admin/orders/[id]` — карточка «Контроль срока» с прогресс-баром
       (`.admin-deadline-card*`, `.admin-deadline-progress*`).
   Цвета берём из admin-токенов (`--admin-{success,warning,danger,*}-*`),
   чтобы не вводить отдельную палитру.
   ========================================================================= */

/* --- KPI-полоса на /admin -------------------------------------------------- */
.admin-deadline-kpis {
  display: grid;
  gap: 0.75rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: var(--admin-radius-card);
  padding: 1rem 1.15rem;
  box-shadow: var(--admin-shadow-soft);
}

.admin-deadline-kpis__header {
  display: grid;
  gap: 0.15rem;
}

.admin-deadline-kpis__title {
  margin: 0;
  font-size: 1rem;
  font-weight: 600;
  color: var(--admin-text);
}

.admin-deadline-kpis__hint {
  margin: 0;
  font-size: 0.85rem;
  color: var(--admin-muted);
}

.admin-deadline-kpis__grid {
  display: grid;
  grid-template-columns: repeat(4, minmax(0, 1fr));
  gap: 0.65rem;
}

@media (max-width: 720px) {
  .admin-deadline-kpis__grid {
    grid-template-columns: repeat(2, minmax(0, 1fr));
  }
}

.admin-deadline-kpi {
  display: grid;
  grid-template-columns: auto 1fr;
  grid-template-rows: auto auto;
  column-gap: 0.6rem;
  align-items: center;
  padding: 0.85rem 1rem;
  border-radius: 14px;
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  color: var(--admin-text);
  text-decoration: none;
  transition: transform 0.12s ease, box-shadow 0.12s ease,
    border-color 0.12s ease;
}
.admin-deadline-kpi:hover {
  transform: translateY(-1px);
  box-shadow: 0 6px 18px rgba(15, 23, 42, 0.08);
  color: inherit;
}
.admin-deadline-kpi:focus-visible {
  outline: 2px solid var(--admin-primary);
  outline-offset: 2px;
}

.admin-deadline-kpi__icon {
  grid-row: 1 / span 2;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 36px;
  height: 36px;
  border-radius: 10px;
  background: var(--admin-surface);
  color: var(--admin-muted);
}

.admin-deadline-kpi__value {
  font-size: 1.35rem;
  font-weight: 700;
  line-height: 1;
}

.admin-deadline-kpi__label {
  font-size: 0.82rem;
  color: var(--admin-muted);
}

.admin-deadline-kpi--danger {
  background: var(--admin-danger-soft);
  border-color: rgba(220, 38, 38, 0.18);
}
.admin-deadline-kpi--danger .admin-deadline-kpi__icon {
  background: rgba(220, 38, 38, 0.12);
  color: var(--admin-danger-fg);
}
.admin-deadline-kpi--danger .admin-deadline-kpi__value {
  color: var(--admin-danger-fg);
}

.admin-deadline-kpi--warning {
  background: var(--admin-warning-soft);
  border-color: rgba(180, 83, 9, 0.18);
}
.admin-deadline-kpi--warning .admin-deadline-kpi__icon {
  background: rgba(180, 83, 9, 0.12);
  color: var(--admin-warning-fg);
}
.admin-deadline-kpi--warning .admin-deadline-kpi__value {
  color: var(--admin-warning-fg);
}

.admin-deadline-kpi--success {
  background: var(--admin-success-soft);
  border-color: rgba(22, 163, 74, 0.18);
}
.admin-deadline-kpi--success .admin-deadline-kpi__icon {
  background: rgba(22, 163, 74, 0.12);
  color: var(--admin-success-fg);
}
.admin-deadline-kpi--success .admin-deadline-kpi__value {
  color: var(--admin-success-fg);
}

.admin-deadline-kpi--muted {
  background: var(--admin-soft);
  border-color: var(--admin-border);
}
.admin-deadline-kpi--muted .admin-deadline-kpi__icon {
  background: var(--admin-surface);
  color: var(--admin-muted);
}

/* --- Табы фильтра на /admin/orders ---------------------------------------- */
.admin-deadline-tabs {
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem;
  align-items: center;
}

.admin-deadline-tab {
  display: inline-flex;
  align-items: center;
  padding: 0.35rem 0.85rem;
  border-radius: var(--admin-radius-pill);
  font-size: 0.85rem;
  font-weight: 500;
  background: var(--admin-soft);
  color: var(--admin-muted);
  border: 1px solid transparent;
  text-decoration: none;
  transition: background 120ms ease, color 120ms ease,
    border-color 120ms ease;
}
.admin-deadline-tab:hover {
  background: var(--admin-surface);
  color: var(--admin-text);
  border-color: var(--admin-border);
}
.admin-deadline-tab--active {
  background: var(--admin-primary-soft);
  color: var(--admin-primary-fg);
  border-color: rgba(37, 99, 235, 0.25);
}
.admin-deadline-tab--overdue.admin-deadline-tab--active {
  background: var(--admin-danger-soft);
  color: var(--admin-danger-fg);
  border-color: rgba(220, 38, 38, 0.25);
}
.admin-deadline-tab--at_risk.admin-deadline-tab--active {
  background: var(--admin-warning-soft);
  color: var(--admin-warning-fg);
  border-color: rgba(180, 83, 9, 0.25);
}
.admin-deadline-tab--on_track.admin-deadline-tab--active {
  background: var(--admin-success-soft);
  color: var(--admin-success-fg);
  border-color: rgba(22, 163, 74, 0.25);
}

/* --- Колонка «Срок» в таблице заказов ------------------------------------- */
.admin-deadline-cell {
  display: grid;
  gap: 0.25rem;
  min-width: 180px;
}

.admin-deadline-cell__row {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  flex-wrap: wrap;
}

.admin-deadline-cell__date {
  font-weight: 500;
  color: var(--admin-text);
}

.admin-deadline-cell__hint {
  color: var(--admin-muted);
  font-size: 0.82rem;
}

/* --- Карточка «Контроль срока» на /admin/orders/[id] ---------------------- */
.admin-deadline-card {
  position: relative;
}

.admin-deadline-card--danger {
  border-color: rgba(220, 38, 38, 0.25);
  background: linear-gradient(
    180deg,
    var(--admin-danger-soft) 0%,
    var(--admin-surface) 70%
  );
}

.admin-deadline-card--warning {
  border-color: rgba(180, 83, 9, 0.25);
  background: linear-gradient(
    180deg,
    var(--admin-warning-soft) 0%,
    var(--admin-surface) 70%
  );
}

/* Прогресс-бар выпуска. Тонкий (6px) и без анимаций — намеренно
   спокойный: задача — показать «сколько готово», а не привлекать
   внимание к самому бару. См. ТЗ §«Не делать: aggressive red blinking». */
.admin-deadline-progress {
  margin-top: 0.45rem;
  width: 100%;
  height: 6px;
  border-radius: 999px;
  background: var(--admin-soft);
  overflow: hidden;
}

.admin-deadline-progress__bar {
  height: 100%;
  background: var(--admin-success);
  border-radius: 999px;
  transition: width 240ms ease;
}

.admin-deadline-card--warning .admin-deadline-progress__bar {
  background: var(--admin-warning);
}

.admin-deadline-card--danger .admin-deadline-progress__bar {
  background: var(--admin-danger);
}

/* --- AdminSizeGrid (компактная сетка размеров в форме заказа) -----------
   Цель: заменить простыню `admin-table` со столбцом «Количество» на
   квадратные плитки. Менеджер видит весь справочник размеров сразу
   (auto-fill 80px), вводит количество прямо в плитку — input стилизован
   как часть карточки, без рамки. См. `admin-size-grid.tsx` для контракта.
   --------------------------------------------------------------------- */

.admin-size-grid {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(80px, 1fr));
  gap: 12px;
}

.admin-size-grid__item {
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  padding: 10px;
  text-align: center;
  display: grid;
  gap: 4px;
  transition: background 160ms ease, border-color 160ms ease,
    box-shadow 160ms ease;
}

.admin-size-grid__item:hover {
  background: var(--admin-primary-soft);
  border-color: #c7d8ff;
}

.admin-size-grid__item:focus-within {
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.15);
  background: var(--admin-surface);
}

.admin-size-grid__label {
  font-size: 0.75rem;
  font-weight: 600;
  color: var(--admin-muted);
  letter-spacing: 0.02em;
  text-transform: uppercase;
}

.admin-size-grid__input {
  width: 100%;
  text-align: center;
  border: none;
  background: transparent;
  font: inherit;
  font-weight: 600;
  font-size: 1rem;
  color: var(--admin-text);
  padding: 2px 0;
  min-width: 0;
  -moz-appearance: textfield;
}

.admin-size-grid__input::-webkit-outer-spin-button,
.admin-size-grid__input::-webkit-inner-spin-button {
  -webkit-appearance: none;
  margin: 0;
}

.admin-size-grid__input:focus {
  outline: none;
}

.admin-size-grid__input:disabled {
  color: var(--admin-text);
  opacity: 1;
  cursor: default;
}

.admin-size-grid__secondary {
  display: flex;
  justify-content: center;
  align-items: baseline;
  gap: 4px;
  font-size: 0.7rem;
  color: var(--admin-muted);
  border-top: 1px dashed var(--admin-border);
  padding-top: 4px;
  margin-top: 2px;
}

.admin-size-grid__secondary-label {
  text-transform: lowercase;
}

.admin-size-grid__secondary-value {
  font-weight: 600;
  color: var(--admin-text);
}

/* =============================================================================
   Admin Order Form 2.0 — карточный layout `/admin/orders/new`
   ----------------------------------------------------------------------------
   Источник правды по структуре — `admin-create-order-form.tsx`. Здесь
   только визуал: pastel-карточки 2x2 + широкий блок «План по размерам»,
   мягкие summary-плашки и нижняя action-панель.

   Палитра pastel-акцентов берётся прямо из ТЗ — это «soft»-фон иконок
   и summary-плашек. Тёмная тема не поддерживается (намеренно: админка
   живёт в светлой палитре, см. начало файла).
   ============================================================================= */

.admin-order-form {
  display: grid;
  gap: var(--admin-space-md);
}

.admin-order-form__error {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.75rem 1rem;
  background: var(--admin-danger-soft);
  border: 1px solid #fecaca;
  border-radius: 14px;
  color: var(--admin-danger-fg);
  font-weight: 500;
}

.admin-order-form__grid {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: var(--admin-space-md);
}

@media (max-width: 900px) {
  .admin-order-form__grid {
    grid-template-columns: 1fr;
  }
}

/* --- Карточка раздела ---------------------------------------------------- */

.admin-order-card {
  background: var(--admin-surface);
  border: 1px solid #dbe3ee;
  border-radius: 20px;
  padding: 1.1rem 1.25rem;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04),
    0 8px 24px rgba(15, 23, 42, 0.04);
  display: grid;
  gap: 0.85rem;
  transition: transform 160ms ease, box-shadow 160ms ease,
    border-color 160ms ease;
}

.admin-order-card:hover {
  transform: translateY(-1px);
  border-color: #c8d4e3;
  box-shadow: 0 2px 4px rgba(15, 23, 42, 0.05),
    0 12px 32px rgba(15, 23, 42, 0.06);
}

.admin-order-card--sizes,
.admin-order-card--full-row {
  grid-column: 1 / -1;
}

.admin-order-card__header {
  display: flex;
  align-items: center;
  gap: 0.65rem;
  min-width: 0;
}

.admin-order-card__icon {
  width: 36px;
  height: 36px;
  border-radius: 12px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  flex: 0 0 auto;
  color: #1e3a8a;
}

.admin-order-card__icon--blue {
  background: #dbeafe;
  color: #1e3a8a;
}

.admin-order-card__icon--green {
  background: #dcfce7;
  color: #166534;
}

.admin-order-card__icon--orange {
  background: #ffedd5;
  color: #9a3412;
}

.admin-order-card__icon--purple {
  background: #ede9fe;
  color: #5b21b6;
}

.admin-order-card__icon--pink {
  background: #f3e8ff;
  color: #6b21a8;
}

.admin-order-card__icon--violet {
  background: #ede9fe;
  color: #5b21b6;
}

.admin-order-card__title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 650;
  letter-spacing: -0.01em;
  color: var(--admin-text);
  line-height: 1.2;
}

/* --- Цветные summary-плашки --------------------------------------------- */

.admin-order-summary {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.75rem;
  padding: 0.7rem 0.95rem;
  border-radius: 14px;
  border: 1px solid transparent;
  background: var(--admin-soft);
  color: var(--admin-text);
  transition: background 160ms ease, border-color 160ms ease,
    color 160ms ease;
}

.admin-order-summary__title {
  font-size: 0.82rem;
  font-weight: 600;
  letter-spacing: 0.02em;
  text-transform: uppercase;
  color: var(--admin-muted);
}

.admin-order-summary__value {
  font-weight: 650;
  font-size: 1rem;
  color: var(--admin-text);
}

.admin-order-summary--blue {
  background: #dbeafe;
  border-color: #c7d8ff;
}
.admin-order-summary--blue .admin-order-summary__title { color: #1e3a8a; }
.admin-order-summary--blue .admin-order-summary__value { color: #1e3a8a; }

.admin-order-summary--green {
  background: #dcfce7;
  border-color: #bbf7d0;
}
.admin-order-summary--green .admin-order-summary__title { color: #166534; }
.admin-order-summary--green .admin-order-summary__value { color: #166534; }

.admin-order-summary--orange {
  background: #ffedd5;
  border-color: #fed7aa;
}
.admin-order-summary--orange .admin-order-summary__title { color: #9a3412; }
.admin-order-summary--orange .admin-order-summary__value { color: #9a3412; }

.admin-order-summary--purple {
  background: #ede9fe;
  border-color: #ddd6fe;
}
.admin-order-summary--purple .admin-order-summary__title { color: #5b21b6; }
.admin-order-summary--purple .admin-order-summary__value { color: #5b21b6; }

.admin-order-summary--warning {
  background: #fef3c7;
  border-color: #fde68a;
}
.admin-order-summary--warning .admin-order-summary__title { color: #92400e; }
.admin-order-summary--warning .admin-order-summary__value { color: #92400e; }

.admin-order-summary--active {
  background: #dcfce7;
  border-color: #bbf7d0;
}
.admin-order-summary--active .admin-order-summary__title { color: #166534; }
.admin-order-summary--active .admin-order-summary__value { color: #166534; }

.admin-order-summary--purple.admin-order-summary--active {
  background: #ede9fe;
  border-color: #c4b5fd;
}
.admin-order-summary--purple.admin-order-summary--active .admin-order-summary__title { color: #5b21b6; }
.admin-order-summary--purple.admin-order-summary--active .admin-order-summary__value { color: #5b21b6; }

.admin-order-summary--warning.admin-order-summary--active {
  background: #dcfce7;
  border-color: #bbf7d0;
}
.admin-order-summary--warning.admin-order-summary--active .admin-order-summary__title { color: #166534; }
.admin-order-summary--warning.admin-order-summary--active .admin-order-summary__value { color: #166534; }

/* --- Preview маршрута --------------------------------------------------- */

.admin-order-route-preview {
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  padding: 0.7rem 0.85rem;
}

/* --- Футер блока «План по размерам» ------------------------------------- */

.admin-order-form__sizes-footer {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: var(--admin-space-md);
  flex-wrap: wrap;
}

.admin-order-form__sizes-footer .admin-order-summary {
  flex: 1 1 220px;
  min-width: 0;
}

.admin-order-form__hint {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  padding: 0.35rem 0.7rem;
  border-radius: 999px;
  background: #dbeafe;
  border: 1px solid #c7d8ff;
  color: #1e3a8a;
  font-size: 0.78rem;
  font-weight: 600;
  letter-spacing: 0.01em;
  white-space: nowrap;
}

/* --- Подсветка активной плитки размера ---------------------------------- */

.admin-size-grid__item--active {
  background: var(--admin-primary-soft);
  border-color: #93c5fd;
  box-shadow: 0 1px 2px rgba(37, 99, 235, 0.08),
    0 4px 12px rgba(37, 99, 235, 0.08);
}

.admin-size-grid__item--active .admin-size-grid__label {
  color: var(--admin-primary-fg);
}

.admin-size-grid__item--active .admin-size-grid__input {
  color: var(--admin-primary-fg);
}

/* --- Нижняя action-панель ----------------------------------------------- */

.admin-order-form__actions {
  display: flex;
  justify-content: flex-end;
  align-items: center;
  gap: var(--admin-space-sm);
  flex-wrap: wrap;
  padding: 0.85rem 1rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 16px;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04),
    0 8px 20px rgba(15, 23, 42, 0.05);
}

@media (max-width: 720px) {
  .admin-order-form__actions {
    justify-content: stretch;
  }
  .admin-order-form__actions .admin-btn {
    flex: 1 1 auto;
    justify-content: center;
  }
}

/* =============================================================================
   Admin Order Form 2.1 — polish-итерация `/admin/orders/new`
   ----------------------------------------------------------------------------
   Дополнительные стили поверх Admin Order Form 2.0:
     - `__top` / `__middle`            — две строки сетки (1+hero, 2+3);
     - `--hero`                        — крупная карточка превью лекала;
     - `.admin-pattern-hero*`          — внутренности hero-карточки;
     - `.admin-order-due-badge`        — мини-плашка «Срок не указан»
                                          внутри карточки заказа;
     - `--secondary` / `__secondary`   — тонкий контейнер для «Учётного
                                          изделия» (старого productId);
     - `.admin-order-pattern-summary`  — read-only сводка лекала
                                          в блоке «Производство»;
     - `--compact`                     — компактный вариант
                                          `.admin-order-summary` для
                                          «Маршрут не выбран».
   ============================================================================= */

/* Верхняя строка — «Заказ» слева, hero-превью справа. На широком экране
   правая колонка чуть уже, чтобы карточка превью оставалась квадратной. */
.admin-order-form__top {
  grid-template-columns: minmax(0, 1.15fr) minmax(0, 1fr);
  align-items: stretch;
}

/* Средняя строка — «Изделие» + «Производство» в равных колонках. */
.admin-order-form__middle {
  grid-template-columns: repeat(2, minmax(0, 1fr));
}

@media (max-width: 900px) {
  .admin-order-form__top,
  .admin-order-form__middle {
    grid-template-columns: 1fr;
  }
}

/* --- Hero-карточка превью лекала ---------------------------------------- */

.admin-order-card--hero {
  gap: 1rem;
}

.admin-pattern-hero {
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
  min-width: 0;
}

.admin-pattern-hero__media {
  position: relative;
  width: 100%;
  height: 240px;
  border-radius: 16px;
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  overflow: hidden;
  display: flex;
  align-items: center;
  justify-content: center;
}

@media (max-width: 900px) {
  .admin-pattern-hero__media {
    height: 220px;
  }
}

.admin-pattern-hero__image {
  width: 100%;
  height: 100%;
  object-fit: contain;
  display: block;
  background: var(--admin-soft);
}

.admin-pattern-hero__placeholder {
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  gap: 0.5rem;
  color: var(--admin-muted);
  font-size: 0.9rem;
  font-weight: 500;
  text-align: center;
  padding: 1rem;
}

.admin-pattern-hero__meta {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  min-width: 0;
}

.admin-pattern-hero__title {
  font-size: 1rem;
  font-weight: 650;
  color: var(--admin-text);
  letter-spacing: -0.005em;
  line-height: 1.25;
  overflow: hidden;
  text-overflow: ellipsis;
}

.admin-pattern-hero__article {
  font-size: 0.85rem;
  font-weight: 500;
  color: var(--admin-muted);
  letter-spacing: 0.02em;
}

.admin-pattern-hero__hint {
  font-size: 0.85rem;
  color: var(--admin-muted);
  line-height: 1.4;
}

/* --- Бейдж «Срок не указан» внутри карточки заказа ---------------------- */

.admin-order-due-badge {
  display: inline-flex;
  align-items: center;
  align-self: flex-start;
  padding: 0.2rem 0.6rem;
  border-radius: 999px;
  background: #fef3c7;
  border: 1px solid #fde68a;
  color: #92400e;
  font-size: 0.75rem;
  font-weight: 600;
  letter-spacing: 0.01em;
  white-space: nowrap;
}

/* --- Read-only summary выбранного лекала в блоке «Производство» --------- */

.admin-order-pattern-summary {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  padding: 0.55rem 0.85rem;
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  border-radius: 12px;
}

.admin-order-pattern-summary__label {
  font-size: 0.72rem;
  font-weight: 600;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--admin-muted);
}

.admin-order-pattern-summary__value {
  font-size: 0.92rem;
  font-weight: 600;
  color: var(--admin-text);
  line-height: 1.3;
  word-break: break-word;
}

/* --- Компактная summary-плашка (для «Маршрут не выбран») --------------- */

.admin-order-summary--compact {
  padding: 0.5rem 0.85rem;
  border-radius: 12px;
}

.admin-order-summary--compact .admin-order-summary__title {
  font-size: 0.7rem;
}

.admin-order-summary--compact .admin-order-summary__value {
  font-size: 0.88rem;
}

/* =============================================================================
   Admin Order Form — компактный «План по размерам» (модалка)
   ----------------------------------------------------------------------------
   Источник правды по логике — `apps/web/app/admin/orders/new/size-plan-selector.tsx`.
   До этой итерации блок «4. План по размерам» рендерил `AdminSizeGrid`
   на ~80 размеров сразу. Теперь по умолчанию рендерится только
   summary («Размеры не выбраны» / «Выбрано размеров: N» + чипсы) и
   одна кнопка `Выбрать размеры` / `Изменить размеры`. Полный список
   инпутов открывается в модалке.

   Кейсы UX (соответствуют `SizePlanSelector`):
     - номенклатура не выбрана — кнопка disabled, подсказка
       «Сначала выберите номенклатуру»;
     - у номенклатуры нет активных DXF — disabled + подсказка
       про карточку номенклатуры;
     - количества не введены — «Размеры не выбраны»;
     - есть выбор — чипсы со списком `S — 20 шт., M — 50 шт. …`.
   ============================================================================= */

/* --- Заголовок карточки с meta-сводкой (Итого: X шт.) ------------------- */

.admin-order-card__header--with-meta {
  flex-wrap: wrap;
  row-gap: 0.35rem;
}

.admin-order-card__header--with-meta .admin-order-card__title {
  flex: 1 1 auto;
  min-width: 0;
}

.admin-order-card__meta {
  display: inline-flex;
  align-items: baseline;
  gap: 0.3rem;
  margin-left: auto;
  padding: 0.2rem 0.65rem;
  border-radius: 999px;
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  color: var(--admin-text);
  font-size: 0.85rem;
  font-weight: 500;
  white-space: nowrap;
  transition: background 160ms ease, border-color 160ms ease,
    color 160ms ease;
}

.admin-order-card__meta--active {
  background: #dcfce7;
  border-color: #bbf7d0;
  color: #166534;
}

.admin-order-card__meta-label {
  color: var(--admin-muted);
  font-weight: 600;
  letter-spacing: 0.01em;
}

.admin-order-card__meta--active .admin-order-card__meta-label {
  color: #166534;
}

.admin-order-card__meta-value {
  font-weight: 700;
}

/* --- Тело блока «План по размерам» -------------------------------------- */

.admin-size-plan {
  display: flex;
  flex-wrap: wrap;
  align-items: flex-start;
  justify-content: space-between;
  gap: 1rem;
}

.admin-size-plan__summary {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
  min-width: 0;
  flex: 1 1 280px;
}

.admin-size-plan__empty-title {
  margin: 0;
  font-size: 1rem;
  font-weight: 600;
  color: var(--admin-text);
  letter-spacing: -0.005em;
}

.admin-size-plan__count {
  margin: 0;
  font-size: 0.95rem;
  color: var(--admin-text);
}

.admin-size-plan__count strong {
  font-weight: 700;
}

.admin-size-plan__hint {
  margin: 0;
  font-size: 0.85rem;
  color: var(--admin-muted);
  line-height: 1.4;
}

.admin-size-plan__hint--muted {
  color: var(--admin-muted);
}

.admin-size-plan__chips {
  list-style: none;
  margin: 0.15rem 0 0;
  padding: 0;
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem;
}

.admin-size-plan__chip {
  display: inline-flex;
  align-items: baseline;
  gap: 0.3rem;
  padding: 0.3rem 0.7rem;
  border-radius: 999px;
  background: var(--admin-primary-soft);
  border: 1px solid #c7d8ff;
  color: var(--admin-primary-fg);
  font-size: 0.85rem;
  font-weight: 500;
  white-space: nowrap;
}

.admin-size-plan__chip-code {
  font-weight: 700;
  letter-spacing: 0.01em;
  text-transform: uppercase;
}

.admin-size-plan__chip-sep {
  opacity: 0.6;
}

.admin-size-plan__chip-qty {
  font-weight: 600;
}

.admin-size-plan__actions {
  display: flex;
  flex: 0 0 auto;
}

.admin-size-plan__open-btn {
  white-space: nowrap;
}

.admin-size-plan__open-btn:disabled {
  opacity: 0.55;
  cursor: not-allowed;
  filter: saturate(0.7);
}

/* --- Модалка размеров --------------------------------------------------- */

.admin-size-plan-modal__backdrop {
  position: fixed;
  inset: 0;
  z-index: 80;
  background: rgba(15, 23, 42, 0.45);
  backdrop-filter: blur(2px);
  display: flex;
  align-items: flex-start;
  justify-content: center;
  padding: 1rem;
  overflow-y: auto;
  overscroll-behavior: contain;
  animation: admin-size-plan-fade 140ms ease-out;
}

@keyframes admin-size-plan-fade {
  from { opacity: 0; }
  to { opacity: 1; }
}

.admin-size-plan-modal {
  width: min(420px, 100%);
  /* `vh`-fallback для Safari < 15.4 (см. `.auth-screen`). */
  max-height: calc(100vh - 2rem);
  max-height: calc(100dvh - 2rem);
  margin: auto;
  background: var(--admin-surface);
  border-radius: 18px;
  border: 1px solid var(--admin-border);
  box-shadow: 0 12px 40px rgba(15, 23, 42, 0.18),
    0 4px 12px rgba(15, 23, 42, 0.08);
  display: flex;
  flex-direction: column;
  overflow: hidden;
  animation: admin-size-plan-pop 160ms cubic-bezier(0.2, 0.8, 0.2, 1);
}

@keyframes admin-size-plan-pop {
  from { transform: translateY(8px) scale(0.98); opacity: 0; }
  to   { transform: translateY(0)   scale(1);    opacity: 1; }
}

.admin-size-plan-modal__header {
  display: flex;
  align-items: center;
  gap: 0.7rem;
  padding: 0.95rem 1.1rem 0.7rem;
  border-bottom: 1px solid var(--admin-border);
  min-width: 0;
}

.admin-size-plan-modal__titles {
  flex: 1 1 auto;
  min-width: 0;
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
}

.admin-size-plan-modal__title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 650;
  color: var(--admin-text);
  letter-spacing: -0.005em;
}

.admin-size-plan-modal__subtitle {
  margin: 0;
  font-size: 0.85rem;
  color: var(--admin-muted);
  line-height: 1.3;
  word-break: break-word;
}

.admin-size-plan-modal__article {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  letter-spacing: 0.01em;
}

.admin-size-plan-modal__close {
  appearance: none;
  background: transparent;
  border: 1px solid transparent;
  border-radius: 10px;
  width: 36px;
  height: 36px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  color: var(--admin-muted);
  cursor: pointer;
  transition: background 140ms ease, color 140ms ease, border-color 140ms ease;
}

.admin-size-plan-modal__close:hover {
  background: var(--admin-soft);
  color: var(--admin-text);
  border-color: var(--admin-border);
}

.admin-size-plan-modal__close:focus-visible {
  outline: none;
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.18);
}

.admin-size-plan-modal__body {
  padding: 0.6rem 1.1rem;
  overflow-y: auto;
  flex: 1 1 auto;
}

.admin-size-plan-modal__rows {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
}

.admin-size-plan-modal__row {
  display: grid;
  grid-template-columns: minmax(80px, 1fr) minmax(120px, 160px);
  align-items: center;
  gap: 0.75rem;
  padding: 0.45rem 0.65rem;
  border-radius: 12px;
  background: var(--admin-soft);
  border: 1px solid var(--admin-border);
  transition: background 140ms ease, border-color 140ms ease;
}

.admin-size-plan-modal__row:hover {
  background: var(--admin-primary-soft);
  border-color: #c7d8ff;
}

.admin-size-plan-modal__row:focus-within {
  background: var(--admin-surface);
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.15);
}

.admin-size-plan-modal__row--active {
  background: var(--admin-primary-soft);
  border-color: #93c5fd;
}

.admin-size-plan-modal__row-label {
  font-size: 0.95rem;
  font-weight: 600;
  color: var(--admin-text);
  letter-spacing: 0.01em;
  text-transform: uppercase;
  cursor: pointer;
  margin: 0;
}

.admin-size-plan-modal__row-input {
  width: 100%;
  text-align: center;
  font: inherit;
  font-size: 1rem;
  font-weight: 600;
  color: var(--admin-text);
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 10px;
  padding: 0.35rem 0.5rem;
  -moz-appearance: textfield;
  transition: border-color 140ms ease, box-shadow 140ms ease;
}

.admin-size-plan-modal__row-input::-webkit-outer-spin-button,
.admin-size-plan-modal__row-input::-webkit-inner-spin-button {
  -webkit-appearance: none;
  margin: 0;
}

.admin-size-plan-modal__row-input:focus {
  outline: none;
  border-color: var(--admin-primary);
  box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.15);
}

.admin-size-plan-modal__footer {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.6rem;
  padding: 0.7rem 1.1rem 0.95rem;
  border-top: 1px solid var(--admin-border);
  background: var(--admin-surface);
  flex-wrap: wrap;
}

.admin-size-plan-modal__total {
  font-size: 0.9rem;
  color: var(--admin-muted);
}

.admin-size-plan-modal__total strong {
  color: var(--admin-text);
  font-weight: 650;
}

.admin-size-plan-modal__footer-actions {
  display: flex;
  align-items: center;
  gap: 0.45rem;
  flex-wrap: wrap;
}

@media (max-width: 520px) {
  .admin-size-plan-modal__row {
    grid-template-columns: 1fr 110px;
  }
  .admin-size-plan-modal__footer {
    align-items: stretch;
  }
  .admin-size-plan-modal__footer-actions {
    width: 100%;
    justify-content: flex-end;
  }
}

/* =============================================================================
   Order Workflow Card (`/admin/orders/[id]`)
   --------------------------------------------------------------------------
   Полировка карточки заказа: выделенный «следующий шаг» сразу под
   header. Источник правды — компонент `OrderWorkflowCard` в
   `apps/web/app/admin/orders/[id]/page.tsx`. По смыслу карточка
   очень похожа на `.admin-deadline-card` (тот же `admin-card` с
   мягким цветным акцентом сверху), но статус-вариантов 5 — поэтому
   живёт отдельным набором классов, чтобы не конфликтовать с
   контролем сроков.
   ========================================================================= */

.admin-order-workflow {
  display: grid;
  gap: 0.75rem;
  position: relative;
  border-left: 4px solid transparent;
}

.admin-order-workflow__head {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  flex-wrap: wrap;
}

.admin-order-workflow__eyebrow {
  font-size: 0.78rem;
  font-weight: 600;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--admin-muted);
}

.admin-order-workflow__body {
  display: grid;
  gap: 0.25rem;
}

.admin-order-workflow__step-label {
  font-size: 0.85rem;
  font-weight: 600;
  color: var(--admin-muted);
}

.admin-order-workflow__hint {
  margin: 0;
  font-size: 1rem;
  line-height: 1.45;
  color: var(--admin-text);
  max-width: 80ch;
}

.admin-order-workflow__actions {
  display: flex;
  flex-wrap: wrap;
  gap: 0.5rem;
}

/* Цветовые тон-варианты карточки. Цвета подобраны под
   `getOrderStatusTone`, чтобы бейдж статуса в шапке карточки
   и фон/полоса слева читались как одно целое. */
.admin-order-workflow--draft {
  background: linear-gradient(
    180deg,
    var(--admin-soft) 0%,
    var(--admin-surface) 70%
  );
  border-left-color: #94a3b8;
}

.admin-order-workflow--calculation {
  background: linear-gradient(
    180deg,
    var(--admin-warning-soft) 0%,
    var(--admin-surface) 70%
  );
  border-left-color: var(--admin-warning);
}

.admin-order-workflow--production {
  background: linear-gradient(
    180deg,
    var(--admin-primary-soft) 0%,
    var(--admin-surface) 70%
  );
  border-left-color: var(--admin-primary);
}

.admin-order-workflow--done {
  background: linear-gradient(
    180deg,
    var(--admin-success-soft) 0%,
    var(--admin-surface) 70%
  );
  border-left-color: var(--admin-success);
}

.admin-order-workflow--cancelled {
  background: linear-gradient(
    180deg,
    var(--admin-danger-soft) 0%,
    var(--admin-surface) 70%
  );
  border-left-color: var(--admin-danger);
}

/* ===========================================================================
   /admin/orders/[id] — единая grid-сетка карточки заказа с
   именованными зонами (`grid-template-areas`).
   ===========================================================================

   Локальный layout только для страницы карточки заказа. Заменяет
   общий `.admin-grid-2` и старую схему двух независимых колонок
   (`admin-order-detail-column--ops` / `--summary`) — при разной
   высоте карточек блоки наезжали друг на друга, потому что
   столбцы выравнивали свои дети независимо. Теперь все 13 блоков
   лежат в одной общей сетке `.admin-order-detail-layout`,
   расставленной по строкам через `grid-template-areas`. Каждый
   ряд (`order | preview`, `deadlines | summary`, `item | plan`,
   …) выравнивается по самому высокому ребёнку, и наезд исчезает.

   Цели CSS-блока:
     - вписать карточку заказа в экран (между sidebar и краем
       экрана) без horizontal overflow body;
     - дать широким таблицам (`MaterialsTable` в «Готовности к
       крою», `PassportsTable`) внутренний скролл вместо того,
       чтобы выталкивать сетку;
     - на узком экране (≤1199px) превращаться в одну колонку
       без потерь порядка блоков.

   Старые классы `.admin-order-detail-grid` / `.admin-order-detail-column`
   оставлены ниже только для обратной совместимости с уже
   существующими местами, которые могут на них ссылаться, но
   layout этой страницы ими больше НЕ управляет. Все новые
   правила — на `admin-order-detail-layout` / `admin-order-area`.

   Никаких изменений `admin-grid-2` / `admin-stack` — те остаются
   такими, как есть, для остальных страниц. Все классы в этом
   блоке начинаются с `admin-order-detail-…` / `admin-order-area`,
   чтобы scope не протекал.
   ----------------------------------------------------------------- */

.admin-order-detail {
  display: grid;
  gap: var(--admin-space-md);
  width: 100%;
  max-width: 100%;
  min-width: 0;
  box-sizing: border-box;
}

/* --- Единая grid-сетка с именованными зонами ---------------------
   Левая и правая колонки одинаковой ширины (`minmax(0, 1fr)`),
   `min-width: 0` у area блокирует растягивание широкими
   таблицами. Каждая строка выравнивает оба своих ребёнка по
   высоте через `align-items: stretch`, поэтому пары
   («Заказ | Превью», «Сроки | Сводка», «Изделие | Готовность
   к крою») получают одну и ту же высоту по эталону — больший
   из двух блоков. Эталон не задаётся фиксированной высотой, а
   растягивается естественно (см. `.admin-order-area > *`).
   ----------------------------------------------------------------- */
.admin-order-detail-layout {
  display: grid;
  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  grid-template-areas:
    'status status'
    'order preview'
    'deadlines summary'
    'production production'
    'item readiness'
    'purchase-orders receipts'
    'passports passports';
  gap: 16px;
  width: 100%;
  max-width: 100%;
  min-width: 0;
  align-items: stretch;
  box-sizing: border-box;
}

.admin-order-area {
  /* Без `min-width: 0` grid-child наследует `min-content` от
     контента, и любая широкая таблица растягивает area за пределы
     экрана. `display: flex` нужен, чтобы единственный ребёнок-карточка
     получил `flex: 1 1 auto` и растянулся по высоте grid-row. */
  min-width: 0;
  box-sizing: border-box;
  display: flex;
}

.admin-order-area > * {
  /* Защита от карточки с фиксированной/слишком большой шириной —
     ограничиваем по ширине ячейки сетки. `flex: 1 1 auto` +
     stretch у grid-row дают одинаковую высоту парных карточек
     (Заказ/Превью, Сроки/Сводка, Изделие/Готовность). */
  width: 100%;
  max-width: 100%;
  flex: 1 1 auto;
}

.admin-order-area--status {
  grid-area: status;
}

.admin-order-area--order {
  grid-area: order;
}

.admin-order-area--preview {
  grid-area: preview;
}

.admin-order-area--deadlines {
  grid-area: deadlines;
}

.admin-order-area--summary {
  grid-area: summary;
}

.admin-order-area--production {
  grid-area: production;
}

.admin-order-area--item {
  grid-area: item;
}

.admin-order-area--readiness {
  grid-area: readiness;
}

.admin-order-area--purchase-orders {
  grid-area: purchase-orders;
}

.admin-order-area--receipts {
  grid-area: receipts;
}

.admin-order-area--passports {
  grid-area: passports;
}

/* `admin-order-detail-card-fill` — явный helper «карточка во всю
   высоту зоны». Не задаёт фиксированную высоту: парные карточки
   («Заказ | Превью», «Сроки | Сводка», «Изделие | Готовность к
   крою») выравниваются по большему блоку через `align-items:
   stretch` родительской grid + `height: 100%` у самой карточки. */
.admin-order-detail-card-fill {
  height: 100%;
}

/* `PatternPreviewCard` живёт в `admin-order-area--preview` и
   должна выровняться по высоте с большим блоком «Заказ» слева.
   Не задаём фиксированную высоту — растягиваем через `height: 100%`
   только в контексте карточки заказа. На других страницах
   (legacy `/orders/[id]`) `.admin-pattern-preview` остаётся
   таким, как есть. */
.admin-order-detail-layout .admin-pattern-preview,
.admin-order-detail-layout .admin-order-area--preview > .admin-card {
  height: 100%;
}

/* `CutReadinessCard` парный блок с «Изделие» — высота тоже
   выравнивается по большему. */
.admin-order-detail-layout .admin-order-area--readiness > .admin-card,
.admin-order-detail-layout .cut-readiness-card {
  height: 100%;
}

/* Компактные карточки только для этой страницы — уменьшаем
   вертикальные/горизонтальные отступы, чтобы блоки не
   расползались по высоте. */
.admin-order-detail-card-compact {
  padding: 0.85rem 1rem;
  gap: 0.5rem;
}

.admin-order-detail-card-compact .admin-section-header {
  margin-bottom: 0.1rem;
}

.admin-order-detail-card-compact .admin-deflist {
  gap: 0.25rem 0.85rem;
}

/* Широкие таблицы внутри карточек (например, MaterialsTable в
   «Готовность к крою» или PassportsTable) не должны выталкивать
   ячейку сетки. `.admin-table-wrap` глобально имеет
   `overflow: hidden`, что обрезает контент; для order detail
   переопределяем на `auto`, чтобы внутри карточки появился
   горизонтальный скролл, а сама страница оставалась в пределах
   экрана. */
.admin-order-detail .admin-table-wrap,
.admin-order-detail-layout .admin-table-wrap {
  overflow-x: auto;
  overflow-y: hidden;
  max-width: 100%;
}

.admin-order-detail .admin-table,
.admin-order-detail-layout .admin-table,
.admin-order-detail-layout table {
  /* Таблица должна жить в потоке, а не растягивать flex/grid
     родителя. Если контент шире — сработает overflow-x выше. */
  width: 100%;
  min-width: 0;
  max-width: 100%;
}

/* Узкий экран: одна колонка, порядок блоков соответствует
   ТЗ (статус → заказ → превью → сроки → сводка → производство
   → изделие → готовность → закупки → поступления → паспорта). */
@media (max-width: 1199px) {
  .admin-order-detail-layout {
    grid-template-columns: minmax(0, 1fr);
    grid-template-areas:
      'status'
      'order'
      'preview'
      'deadlines'
      'summary'
      'production'
      'item'
      'readiness'
      'purchase-orders'
      'receipts'
      'passports';
  }
}

/* ---------------------------------------------------------------------------
   /admin/orders/[id] — объединённый блок «Изделие».
   ---------------------------------------------------------------------------

   Один общий блок «Изделие» содержит три логические подсекции:
     A) краткая информация (название / цвет / артикул лекала);
     B) план по размерам (read-only `OrderItemsGrid` + chips размеров);
     C) потребность цеха (вложенный `WorkshopNeedsCard`).

   Backend / DTO / `WorkshopNeedsService` не меняли — компонент
   просто отрендерен внутри родительской карточки. Чтобы не
   получалось «карточка в карточке с двойными бордерами», у
   вложенного `.admin-card` отключаем фон/бордеры/тень/паддинг —
   `WorkshopNeedsCard` визуально сливается с родителем.
   --------------------------------------------------------------------------- */
.admin-order-item-card {
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
}

.admin-order-item-card__section {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
  border-top: 1px solid var(--admin-border);
  padding-top: 0.7rem;
  min-width: 0;
}

.admin-order-item-card__subhead {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 0.6rem;
  flex-wrap: wrap;
}

.admin-order-item-card__subtitle {
  margin: 0;
  font-size: 0.78rem;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  color: var(--admin-muted);
}

.admin-order-item-card__meta {
  font-size: 0.85rem;
  color: var(--admin-muted);
}

/* Бейдж источника номенклатуры в блоке «Изделие». Показывается
   только когда `resolveOrderNomenclature` опустился до legacy
   `productName` — так менеджер сразу видит, что заказ исторический,
   без карточки лекала. Для snapshot/live PatternItem бейдж тут не
   нужен: в этой карточке снимок от живого PatternItem-а уже
   неотличим — оба источника указывают на одно и то же лекало,
   просто на разный его «слой времени» (и соответствующий бейдж
   живёт в `PatternPreviewCard` сверху). */
.admin-order-item-card__source-badge {
  display: inline-block;
  margin-left: 0.4rem;
  padding: 1px 6px;
  border-radius: 999px;
  background: rgba(148, 163, 184, 0.18);
  color: var(--admin-muted, rgba(0, 0, 0, 0.55));
  font-size: 0.7rem;
  font-weight: 500;
  vertical-align: middle;
}

/* Вложенный `WorkshopNeedsCard` теряет визуал отдельной карточки —
   её роль играет родительский «Изделие». Сохраняем grid+gap у
   `.admin-card` (это даёт нормальные отступы между AdminSectionHeader
   / AdminEmptyState / списком потребностей внутри). */
.admin-order-item-card .admin-card {
  background: transparent;
  border: none;
  box-shadow: none;
  padding: 0;
}

/* Этап «Нанесение на заказе покупателя»: read-only список нанесений
   в блоке «Изделие» (компонент `OrderApplicationsCard`). Form-mode
   (DRAFT) использует inline-стили в самой форме; здесь — только стиль
   read-only списка. */
.admin-order-applications__list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
}

.admin-order-applications__item {
  border: 1px solid var(--admin-border);
  border-radius: 6px;
  padding: 0.6rem 0.8rem;
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
}

.admin-order-applications__item-head {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 0.5rem;
  font-size: 0.92rem;
}

.admin-deflist--compact dt {
  font-size: 0.78rem;
}

.admin-deflist--compact dd {
  font-size: 0.85rem;
}

/* Этап «Нанесение по размерам / части тиража»: блок «Применить к»
   в редакторе нанесения (весь тираж / выбранные размеры). Редактор
   живёт в client-компоненте с inline-стилями полей; здесь — только
   стиль адресации по размерам, чтобы чипы размеров и поразмерное
   количество выглядели аккуратно. */
.admin-order-applications__scope {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
  padding: 0.5rem 0.6rem;
  border: 1px dashed var(--admin-border, #e5e7eb);
  border-radius: 6px;
  background: var(--admin-surface-subtle, #fafafa);
}

.admin-order-applications__scope-label {
  font-size: 0.78rem;
  font-weight: 600;
  color: var(--admin-muted, #6b7280);
}

.admin-order-applications__scope-modes {
  display: flex;
  flex-wrap: wrap;
  gap: 0.9rem;
}

.admin-order-applications__radio {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  font-size: 0.85rem;
  cursor: pointer;
}

.admin-order-applications__sizes {
  display: flex;
  flex-wrap: wrap;
  gap: 0.5rem;
}

.admin-order-applications__size-chip {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.25rem 0.5rem;
  border: 1px solid var(--admin-border, #e5e7eb);
  border-radius: 999px;
  background: var(--admin-surface, #fff);
}

.admin-order-applications__size-chip--on {
  border-color: var(--admin-accent, #6366f1);
  background: var(--admin-accent-soft, #eef2ff);
}

.admin-order-applications__size-toggle {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  font-size: 0.85rem;
  cursor: pointer;
}

.admin-order-applications__size-qty {
  width: 4.5rem;
  font-size: 0.8rem;
  padding: 0.1rem 0.35rem;
}

/* Карточка одиночного нанесения в редакторе (раньше — inline-border). */
.admin-order-applications__row {
  border: 1px solid var(--admin-border, #e5e7eb);
  border-radius: 6px;
  padding: 12px;
  display: flex;
  flex-direction: column;
  gap: 8px;
}

/* Этап «Комплекты нанесений»: карточка комплекта в редакторе. */
.admin-order-applications__kit {
  border: 1px solid var(--admin-accent, #6366f1);
  border-radius: 8px;
  padding: 12px;
  display: flex;
  flex-direction: column;
  gap: 10px;
  background: var(--admin-accent-soft, #eef2ff);
}

.admin-order-applications__kit-head {
  display: flex;
  align-items: center;
  gap: 8px;
  flex-wrap: wrap;
}

.admin-order-applications__kit-name {
  flex: 1 1 240px;
  font-weight: 600;
  font-size: 0.95rem;
  padding: 0.3rem 0.5rem;
}

.admin-order-applications__kit-members {
  display: flex;
  flex-direction: column;
  gap: 8px;
}

.admin-order-applications__kit-member {
  border: 1px solid var(--admin-border, #e5e7eb);
  border-radius: 6px;
  padding: 10px;
  display: flex;
  flex-direction: column;
  gap: 8px;
  background: var(--admin-surface, #fff);
}

.admin-order-applications__kit-actions {
  display: flex;
  flex-wrap: wrap;
  gap: 8px;
}

/* Read-only: группировка нанесений по комплектам в карточке заказа. */
.admin-order-applications__blocks {
  display: flex;
  flex-direction: column;
  gap: 0.7rem;
}

.admin-order-applications__kit-ro {
  border: 1px solid var(--admin-accent, #6366f1);
  border-radius: 8px;
  padding: 0.5rem 0.7rem;
  background: var(--admin-accent-soft, #eef2ff);
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}

.admin-order-applications__kit-ro-head {
  display: flex;
  align-items: center;
  gap: 0.4rem;
  font-size: 0.92rem;
}

.admin-order-applications__kit-ro-scope {
  margin-left: auto;
  font-size: 0.8rem;
  color: var(--admin-muted, #6b7280);
}

/* ---------------------------------------------------------------------------
   /admin/orders/[id] — блок «Готовность к крою»: компактные
   статусы и горизонтальный скролл таблицы материалов.
   ---------------------------------------------------------------------------

   Раньше:
     - статусы (`Блокеров / Предупреждений / Проверено`) рисовались
       через `.admin-deflist`, и при узкой grid-area колонки
       разъезжались / переносились вразнобой;
     - таблица материалов внутри `.admin-table-wrap` имела
       `width: 100%; min-width: 0`, и контент сжимался до того,
       что часть колонок выходила за карточку.

   Стало:
     - стат-боксы лежат в `auto-fit grid` — на узкой ширине
       сворачиваются в одну колонку, на широкой — в три;
     - таблица материалов получает `min-width: 720px` и живёт в
       `overflow-x: auto`-обёртке, поэтому скроллится внутри
       блока, а не выталкивает grid-сетку страницы.
   --------------------------------------------------------------------------- */
.cut-readiness-card {
  min-width: 0;
}

.cut-readiness-stats {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(120px, 1fr));
  gap: 8px;
  margin-top: 4px;
}

.cut-readiness-stats__item {
  min-width: 0;
  display: flex;
  flex-direction: column;
  gap: 2px;
  padding: 8px 10px;
  border: 1px solid var(--admin-border);
  border-radius: 8px;
  background: var(--admin-surface);
}

.cut-readiness-stats__label {
  font-size: 0.72rem;
  color: var(--admin-muted);
  text-transform: uppercase;
  letter-spacing: 0.04em;
}

.cut-readiness-stats__value {
  font-size: 0.95rem;
  font-weight: 600;
  word-break: break-word;
}

.cut-readiness-section {
  margin-top: 12px;
  min-width: 0;
}

.cut-readiness-section__title {
  margin: 0 0 6px 0;
  font-size: 0.82rem;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  color: var(--admin-muted);
}

.cut-readiness-list {
  margin: 0;
  padding: 0;
  list-style: none;
}

.cut-readiness-list__item {
  padding: 6px 0;
  border-top: 1px solid var(--admin-border);
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 12px;
  word-break: break-word;
  overflow-wrap: anywhere;
}

.cut-readiness-list__body {
  min-width: 0;
  flex: 1 1;
  word-break: break-word;
  overflow-wrap: anywhere;
}

.cut-readiness-list__title {
  font-weight: 500;
}

.cut-readiness-list__msg {
  font-size: 0.78rem;
  margin-top: 2px;
}

.cut-readiness-materials-scroll {
  width: 100%;
  max-width: 100%;
  overflow-x: auto;
  overflow-y: hidden;
  -webkit-overflow-scrolling: touch;
}

/* Внутри блока «Готовность к крою» таблица материалов имеет
   собственный min-width — длинные описания материалов и список
   ячеек не должны ужиматься в нечитаемую колонку. Скролл
   обеспечивает обёртка `.cut-readiness-materials-scroll`. */
.cut-readiness-materials-scroll .admin-table-wrap {
  border: none;
  box-shadow: none;
  background: transparent;
  border-radius: 0;
  overflow: visible;
}

.cut-readiness-materials-scroll .admin-table {
  min-width: 720px;
  width: max-content;
  max-width: none;
}

.cut-readiness-materials-scroll .admin-table td,
.cut-readiness-materials-scroll .admin-table th {
  word-break: break-word;
  overflow-wrap: anywhere;
}

.cut-readiness-materials-scroll .admin-table td.is-numeric,
.cut-readiness-materials-scroll .admin-table th.is-numeric {
  white-space: nowrap;
}

/* ---------------------------------------------------------------------------
   /admin/orders/[id] — иконка «Контроль выпуска» в шапке «Сводки».
   ---------------------------------------------------------------------------

   Из тела карточки «Сводка» убрали строки «Создан» (createdAt) и
   «Контроль срока». Чтобы не терять информацию о deadline tone,
   оставляем компактную иконку в правом верхнем углу секции —
   она цвет наследует от admin-status-tone (см. AdminStatusBadge),
   а сама занимает один-в-один ширину иконки + 2px паддинга.
   --------------------------------------------------------------------------- */
.admin-order-summary-control {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 28px;
  height: 28px;
  border-radius: 999px;
  border: 1px solid currentColor;
  color: var(--admin-muted);
  background: transparent;
  flex-shrink: 0;
}

.admin-order-summary-control--success {
  color: var(--admin-success, #15803d);
}

.admin-order-summary-control--info {
  color: var(--admin-info, #2563eb);
}

.admin-order-summary-control--warning {
  color: var(--admin-warning, #b45309);
}

.admin-order-summary-control--danger {
  color: var(--admin-danger, #b91c1c);
}

/* ---------------------------------------------------------------------------
   /admin/orders/[id] — пояснение под карточкой «Производство».
   --------------------------------------------------------------------------- */
.admin-order-production-note {
  margin: 0.4rem 0 0 0;
  font-size: 0.82rem;
  color: var(--admin-muted);
}

/* Этап 2 «План операций на заказе» (см.
   `docs/operation-time-norms-recon.md §10/§11`).
   Компактный блок «План операций» внутри карточки «Производство»
   на `/admin/orders/[id]`. Визуально сливается с родительской
   карточкой — без рамки, лёгкий разделитель сверху, чтобы не
   рисовать «карточку в карточке». Warnings — через subdued
   акцент с иконкой; список ужат, остальные «ещё N». */
.admin-order-operation-plan {
  margin-top: 0.7rem;
  padding-top: 0.7rem;
  border-top: 1px solid var(--admin-border);
}
.admin-order-operation-plan__title {
  font-size: 0.85rem;
  font-weight: 600;
  margin-bottom: 0.4rem;
}
.admin-order-operation-plan__empty {
  font-size: 0.84rem;
}
.admin-order-operation-plan__list {
  margin: 0;
}
.admin-order-operation-plan__warnings {
  margin-top: 0.5rem;
  padding: 0.45rem 0.6rem;
  border-radius: 6px;
  background: var(--admin-warning-bg, #fff7ed);
  color: var(--admin-warning-fg, #9a3412);
  font-size: 0.82rem;
}
.admin-order-operation-plan__warnings-title {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  font-weight: 600;
  margin-bottom: 0.25rem;
}
.admin-order-operation-plan__warnings-list {
  margin: 0;
  padding-left: 1rem;
  list-style: disc;
}
.admin-order-operation-plan__warnings-list li {
  margin: 0.1rem 0;
}

/* Этап 2 «План операций на заказе» — UI-расширение этапа 2 (см.
   ТЗ «Доработать план операций в заказе»):
     - badge «Требует пересчёта» (staleness),
     - подсказки для CALCULATION_DONE / IN_PRODUCTION,
     - блок управления (кнопка «Пересчитать план операций»).

   Стиль badge синхронен с `.admin-order-operation-plan__warnings`,
   но цвет тревожнее — это явный action-required сигнал, а warnings
   читаются как «справочно». */
.admin-order-operation-plan__stale {
  margin-bottom: 0.55rem;
  padding: 0.45rem 0.6rem;
  border-radius: 6px;
  background: var(--admin-warning-bg, #fff7ed);
  color: var(--admin-warning-fg, #9a3412);
  font-size: 0.82rem;
  border: 1px solid var(--admin-warning-fg, #9a3412);
}
.admin-order-operation-plan__stale-title {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  font-weight: 600;
  margin-bottom: 0.2rem;
}
.admin-order-operation-plan__stale-reason {
  font-size: 0.8rem;
  line-height: 1.35;
}

.admin-order-operation-plan__actions {
  margin-top: 0.6rem;
}
.admin-order-operation-plan__hint {
  font-size: 0.82rem;
  font-style: italic;
}

/* ---------------------------------------------------------------------------
   /admin/orders/[id] — пара «План операций | Плановая себестоимость»
   (см. `apps/web/components/orders/order-planned-cost-summary-card.tsx`).

   Внутренняя сетка для двух компактных блоков в карточке «4.
   Производство». Слева — `OrderOperationPlanBlock` (стоимость и время
   операций), справа — `OrderPlannedCostSummaryCard` (материалы /
   фурнитура / нанесение / операции / итого + «за 1 изделие»).

   На десктопе две колонки одинаковой ширины (`minmax(0, 1fr)` чтобы
   контент не растягивал ячейку). Обе карточки сами рисуют свой
   разделитель сверху (border-top), поэтому общего фона у грида нет —
   он только раскладывает блоки в строку. На узком экране (≤1199px)
   — одна колонка.
   --------------------------------------------------------------------------- */
.admin-order-production-cost-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  gap: 0.9rem;
  align-items: stretch;
}
.admin-order-production-cost-grid > * {
  min-width: 0;
}
@media (max-width: 1199px) {
  .admin-order-production-cost-grid {
    grid-template-columns: minmax(0, 1fr);
    gap: 0.6rem;
  }
}

/* ---------------------------------------------------------------------------
   /admin/orders/[id] — блок «Плановая себестоимость»
   (см. `apps/web/components/orders/order-planned-cost-summary-card.tsx`).

   Компактная finance-summary в стиле SaaS-дашборда:
     - заголовок с иконкой;
     - подсказка-источник («По завершённому расчёту» / «Предварительно
       по заполненным ценам потребности» / «Заполните цены в потребности
       цеха»);
     - список строк kind → сумма;
     - вспомогательные строки «за 1 изделие» — приглушённого тона,
       чтобы итог читался первым;
     - две строки итого (сверху разделитель), слегка крупнее;
     - warning-бейдж для USD и для ошибок загрузки.

   Цветовая схема и border-сверху повторяет стиль соседнего блока
   «План операций» (`.admin-order-operation-plan`), чтобы пара
   читалась как «два листа одного бланка».
   --------------------------------------------------------------------------- */
.order-planned-cost-card {
  margin-top: 0.7rem;
  padding-top: 0.7rem;
  border-top: 1px solid var(--admin-border);
  display: flex;
  flex-direction: column;
  gap: 0.45rem;
}
.order-planned-cost-card__head {
  display: flex;
  align-items: flex-start;
  gap: 0.45rem;
}
.order-planned-cost-card__head-icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  flex: 0 0 auto;
  width: 22px;
  height: 22px;
  border-radius: 6px;
  background: var(--admin-info-bg, #eef2ff);
  color: var(--admin-info-fg, #4338ca);
}
.order-planned-cost-card__head-text {
  display: flex;
  flex-direction: column;
  min-width: 0;
}
.order-planned-cost-card__title {
  font-size: 0.85rem;
  font-weight: 600;
  line-height: 1.2;
}
.order-planned-cost-card__hint {
  font-size: 0.78rem;
  line-height: 1.3;
}
.order-planned-cost-card__empty {
  font-size: 0.84rem;
  padding: 0.2rem 0;
}
.order-planned-cost-card__rows {
  margin: 0;
  display: flex;
  flex-direction: column;
  gap: 0.18rem;
}
.order-planned-cost-card__row {
  display: grid;
  grid-template-columns: minmax(0, 1fr) auto;
  align-items: baseline;
  gap: 0.5rem;
  padding: 0.12rem 0;
}
.order-planned-cost-card__label {
  margin: 0;
  font-size: 0.85rem;
  color: var(--admin-text);
}
.order-planned-cost-card__value {
  margin: 0;
  font-size: 0.85rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  text-align: right;
  white-space: nowrap;
}
.order-planned-cost-card__row--muted .order-planned-cost-card__label,
.order-planned-cost-card__row--muted .order-planned-cost-card__value {
  color: var(--admin-muted-fg, #6b7280);
  font-size: 0.8rem;
}
.order-planned-cost-card__row--total {
  border-top: 1px solid var(--admin-border);
  padding-top: 0.35rem;
  margin-top: 0.18rem;
}
.order-planned-cost-card__row--total .order-planned-cost-card__label,
.order-planned-cost-card__row--total .order-planned-cost-card__value {
  font-weight: 600;
  font-size: 0.92rem;
  color: var(--admin-text);
}
.order-planned-cost-card__ops-label {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  flex-wrap: wrap;
}
.order-planned-cost-card__stale-badge {
  display: inline-flex;
  align-items: center;
  font-size: 0.7rem;
  line-height: 1;
  padding: 0.18rem 0.4rem;
  border-radius: 999px;
  background: var(--admin-warning-bg, #fff7ed);
  color: var(--admin-warning-fg, #9a3412);
  border: 1px solid var(--admin-warning-fg, #9a3412);
  font-weight: 600;
  letter-spacing: 0.01em;
}
.order-planned-cost-card__warning {
  display: flex;
  align-items: flex-start;
  gap: 0.35rem;
  margin-top: 0.2rem;
  padding: 0.45rem 0.6rem;
  border-radius: 6px;
  background: var(--admin-warning-bg, #fff7ed);
  color: var(--admin-warning-fg, #9a3412);
  font-size: 0.8rem;
  line-height: 1.35;
}

/* ---------------------------------------------------------------------------
   /admin/orders/[id] — блок «Производственная цепочка»
   (см. `apps/web/components/orders/production-balance-card.tsx`,
   `apps/api/src/modules/orders/order-production-balance.service.ts`).

   Сетка статов 2×2 на десктопе, 1×4 на мобильном через
   `repeat(auto-fit, minmax(...))`. Чипсы «Раскройщик 1 · Оверлок 2»
   — flex-row с переносом, без явных bullet-ов. Таблица
   балансировки — стандартная `AdminTable`, обёрнутая в
   `overflow-x: auto` контейнер, чтобы при узкой ширине карточка
   страницы не растягивалась.
   --------------------------------------------------------------------------- */
.admin-order-production-balance__summary {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(160px, 1fr));
  gap: 0.6rem;
  margin-top: 0.6rem;
}
.admin-order-production-balance__stat {
  border: 1px solid var(--admin-border);
  border-radius: 8px;
  padding: 0.55rem 0.7rem;
  background: var(--admin-bg-soft, #fafafa);
}
.admin-order-production-balance__stat--warning {
  border-color: rgba(243, 156, 18, 0.45);
  background: rgba(243, 156, 18, 0.08);
}
.admin-order-production-balance__stat--success {
  border-color: rgba(46, 204, 113, 0.45);
  background: rgba(46, 204, 113, 0.08);
}
.admin-order-production-balance__stat-label {
  font-size: 0.78rem;
  color: var(--admin-muted, #666);
  margin-bottom: 0.2rem;
}
.admin-order-production-balance__stat-value {
  font-size: 1rem;
  font-weight: 600;
}
.admin-order-production-balance__staffing {
  margin-top: 0.7rem;
  padding-top: 0.6rem;
  border-top: 1px dashed var(--admin-border);
}
.admin-order-production-balance__staffing-title {
  font-size: 0.85rem;
  font-weight: 600;
  margin-bottom: 0.35rem;
}
.admin-order-production-balance__staffing-row {
  display: flex;
  flex-wrap: wrap;
  gap: 0.35rem 0.6rem;
}
.admin-order-production-balance__staffing-chip {
  display: inline-flex;
  align-items: baseline;
  gap: 0.3rem;
  padding: 0.2rem 0.55rem;
  border-radius: 999px;
  background: var(--admin-bg-soft, #fafafa);
  border: 1px solid var(--admin-border);
  font-size: 0.85rem;
}
.admin-order-production-balance__recommendation {
  margin-top: 0.7rem;
  padding: 0.55rem 0.75rem;
  border-radius: 8px;
  background: rgba(46, 204, 113, 0.06);
  border: 1px solid rgba(46, 204, 113, 0.35);
}
.admin-order-production-balance__recommendation-title {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  font-size: 0.85rem;
  font-weight: 600;
  margin-bottom: 0.3rem;
}
.admin-order-production-balance__recommendation-body {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
}
.admin-order-production-balance__table-wrap {
  margin-top: 0.7rem;
  overflow-x: auto;
}
.admin-order-production-balance__warnings {
  margin-top: 0.6rem;
  padding: 0.45rem 0.6rem;
  border-radius: 6px;
  background: rgba(243, 156, 18, 0.08);
  border: 1px solid rgba(243, 156, 18, 0.35);
}
.admin-order-production-balance__warnings-title {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  font-size: 0.83rem;
  font-weight: 600;
  margin-bottom: 0.25rem;
}
.admin-order-production-balance__warnings-list {
  margin: 0;
  padding-left: 1rem;
  list-style: disc;
  font-size: 0.82rem;
}
.admin-order-production-balance__warnings-list li {
  margin: 0.1rem 0;
}

/* --- Совместимость со старой схемой (deprecated) -----------------
   Классы `.admin-order-detail-grid` / `.admin-order-detail-column`
   больше не используются как основной layout страницы. Оставляем
   минимальные стили на случай ссылок из тестов/legacy-кода, но
   они не должны управлять раскладкой `/admin/orders/[id]`.
   ----------------------------------------------------------------- */
.admin-order-detail-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  gap: 16px;
  width: 100%;
  max-width: 100%;
  min-width: 0;
}

.admin-order-detail-column {
  display: grid;
  gap: 12px;
  min-width: 0;
}

/* ---------------------------------------------------------------------------
   /admin/patterns — иконка категории номенклатуры.

   Используется тремя UI-местами:
     - `.pattern-category-icon`            — строка таблицы / списка (32×32);
     - `.pattern-category-icon--filter`    — chip-фильтр сверху страницы
       (плитка 80×80 без текста — название категории
       вынесено в `title`/`aria-label` ссылки `CategoryFilterChip`);
     - `.pattern-category-icon--detail`    — карточка `/admin/patterns/[id]` (48×48).

   Внутри контейнера может быть либо `<img>` (загруженный JPG/PNG), либо
   lucide-`<svg>` fallback (см. `apps/web/lib/pattern-category-icon.tsx`).
   `object-fit: cover` + одинаковый радиус делают чипсы визуально однородными
   независимо от того, что грузил менеджер (фото / графика / icon).
   --------------------------------------------------------------------------- */
.pattern-category-icon {
  width: 32px;
  height: 32px;
  border-radius: 10px;
  overflow: hidden;
  flex: 0 0 auto;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  background: var(--admin-soft, #f1f5f9);
  border: 1px solid var(--admin-border, #e2e8f0);
  color: var(--admin-muted, #64748b);
}

.pattern-category-icon--filter {
  width: 80px;
  height: 80px;
  border-radius: 16px;
}

.pattern-category-icon--detail {
  width: 48px;
  height: 48px;
  border-radius: 14px;
}

.pattern-category-icon img {
  width: 100%;
  height: 100%;
  object-fit: cover;
  display: block;
}

.pattern-category-icon svg {
  width: 55%;
  height: 55%;
}

/* В плитке-фильтре fallback-svg растягиваем сильнее, чтобы
   иконка не «терялась» на фоне 80×80 квадрата. */
.pattern-category-icon--filter svg {
  width: 60%;
  height: 60%;
}

/* ---------------------------------------------------------------------------
   /admin/patterns — редактируемый чип-фильтр категории.

   Wrapper `.pattern-category-filter` оборачивает два sibling-link-а:
     - основной `<Link>` (фильтр списка лекал по `categoryId`);
     - небольшая `pattern-category-filter__edit` (ссылка на
       `/admin/pattern-categories/<id>`), которая появляется только
       при hover родителя или focus-visible внутри.

   Wrapper выбран вместо вложенного `<a>` сознательно: в HTML нельзя
   вкладывать ссылку в ссылку, поэтому edit-кнопка лежит «поверх» в
   правом верхнем углу через `position: absolute`.
   --------------------------------------------------------------------------- */
.pattern-category-filter {
  position: relative;
  display: inline-flex;
}

/* Текстовый label категории внутри чипа-фильтра (этап «Подписать
   группы номенклатуры»). Иконка остаётся слева через `gap` на самой
   ссылке; запас справа — чтобы кластер действий не перекрывал текст. */
.pattern-category-filter__label {
  font-weight: 600;
  white-space: nowrap;
}

.pattern-category-filter .admin-btn {
  padding-right: 14px;
}

/* Приглушённый вид для архивных категорий. */
.pattern-category-filter__link--archived {
  opacity: 0.6;
}

.pattern-category-filter__link--archived:hover {
  opacity: 0.85;
}

/* Кластер действий (карандаш + корзина) в правом-верхнем углу чипа.
   Скрыт по умолчанию, появляется на hover/focus-within родителя. */
.pattern-category-filter__actions {
  position: absolute;
  top: -8px;
  right: -8px;
  display: inline-flex;
  gap: 4px;
  opacity: 0;
  pointer-events: none;
  transition: opacity 120ms ease;
  z-index: 2;
}

.pattern-category-filter:hover .pattern-category-filter__actions,
.pattern-category-filter:focus-within .pattern-category-filter__actions {
  opacity: 1;
  pointer-events: auto;
}

.pattern-category-filter__act {
  width: 22px;
  height: 22px;
  border-radius: 50%;
  background: var(--admin-card, #ffffff);
  border: 1px solid var(--admin-border, #e2e8f0);
  color: var(--admin-text, #1e293b);
  display: inline-flex;
  align-items: center;
  justify-content: center;
  transition: transform 120ms ease, background 120ms ease,
    border-color 120ms ease;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.08);
  padding: 0;
  text-decoration: none;
  cursor: pointer;
}

.pattern-category-filter__act:hover {
  background: var(--admin-soft, #f1f5f9);
  transform: scale(1.05);
}

.pattern-category-filter__act:focus-visible {
  outline: 2px solid var(--admin-accent, #2563eb);
  outline-offset: 2px;
}

.pattern-category-filter__act:disabled {
  opacity: 0.6;
  cursor: progress;
}

.pattern-category-filter__act--danger {
  color: var(--admin-danger, #dc2626);
  border-color: var(--admin-danger-border, #fecaca);
}

.pattern-category-filter__act--danger:hover {
  background: var(--admin-danger-soft, #fef2f2);
  border-color: var(--admin-danger, #dc2626);
}

.pattern-category-filter__act svg {
  width: 12px;
  height: 12px;
}


/* ---------------------------------------------------------------------------
   /admin/workshop-needs — polish-итерация «По заказам / Построчно».
   ---------------------------------------------------------------------------

   Polish-итерация раздела «Потребность цеха». Расчёт WorkshopNeed
   и Prisma не менялись — это исключительно UI-стили.

   Базовые принципы:
     - карточка заказа должна быть компактной (одна строка header
       умещает превью + meta + статус), без огромных полей;
     - каждая секция «Материалы / Фурнигура / Нанесение / Прочее»
       рисуется одним вертикальным списком (`workshop-need-section`);
     - бейдж типа (Material / Hardware / Application / Other)
       цветовой, но мягкий — не перетягивает внимание со статуса.

   Все цвета — через CSS-переменные `--admin-*` для согласованности
   с остальными админ-страницами; в dark theme переменные перекрываются
   глобально (см. начало файла).
   --------------------------------------------------------------------------- */

/* `.workshop-needs-view-toggle` наследует общий `admin-tabs`-стиль —
   отдельные правила не нужны, маркер-класс используется только для
   smoke-тестов и ручного DOM-инспектирования. */

.workshop-order-group-list {
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
}

/* SaaS-итерация «Карточка заказа в view=orders».
   Карточка — белый прямоугольник, padding 16–20px, border 12–14px.
   Header: identity (preview + meta + stats) слева, actions
   (кнопки + complete-calculation-form) справа. Body — компактные
   секции «Материалы / Фурнитура / Нанесение / Прочее». Внизу
   карточки никакого «footer» нет — все действия (Открыть заказ,
   Завершить расчёт) живут в правом верхнем углу. */
.workshop-order-group-card {
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
  border: 1px solid var(--admin-border, #e3e9f2);
  border-radius: 14px;
  background: var(--admin-card-bg, #fff);
  padding: 18px 20px;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04);
}

.workshop-order-group-card__header {
  display: flex;
  align-items: flex-start;
  gap: 1rem;
  justify-content: space-between;
  flex-wrap: wrap;
  padding-bottom: 14px;
  border-bottom: 1px solid var(--admin-border, #e3e9f2);
}

.workshop-order-group-card__identity {
  display: flex;
  align-items: flex-start;
  gap: 0.85rem;
  min-width: 0;
  flex: 1 1 320px;
}

.workshop-order-group-card__preview {
  flex-shrink: 0;
}

.workshop-order-group-card__info {
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
  min-width: 0;
  flex: 1 1 auto;
}

.workshop-order-group-card__meta {
  display: flex;
  flex-direction: column;
  gap: 0.25rem;
  min-width: 0;
  flex: 1 1 auto;
}

.workshop-order-group-card__title-row {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  flex-wrap: wrap;
}

.workshop-order-group-card__order-number {
  font-size: 1.05rem;
  font-weight: 600;
}

.workshop-order-group-card__meta-line {
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem 0.85rem;
  color: var(--admin-text-muted, #475569);
  font-size: 0.85rem;
  min-width: 0;
}

.workshop-order-group-card__meta-item {
  display: inline-flex;
  align-items: center;
  gap: 0.25rem;
}

.workshop-order-group-card__article {
  font-size: 0.78rem;
  background: var(--admin-soft, #f1f5f9);
  border-radius: 4px;
  padding: 1px 6px;
}

.workshop-order-group-card__stats {
  display: flex;
  flex-wrap: wrap;
  gap: 0.35rem 0.85rem;
  font-size: 0.78rem;
  color: var(--admin-text-muted, #475569);
  margin-top: 0.15rem;
}

.workshop-order-group-card__stat strong {
  color: var(--admin-text, #1e293b);
}

.workshop-order-group-card__actions {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 0.5rem;
  flex-shrink: 0;
}

.workshop-order-group-card__action {
  white-space: nowrap;
}

.workshop-order-group-card__body {
  display: flex;
  flex-direction: column;
  gap: 12px;
}

/* Превью изделия в обоих режимах (sm — таблица 40×40, md — карточка 88×88).
   Используем `object-fit: contain`, чтобы изделие никогда не обрезалось.
   Placeholder появляется, если `nomenclaturePreviewImageUrl` пуст. */
.workshop-order-preview {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  border-radius: 8px;
  border: 1px solid var(--admin-border, #e2e8f0);
  background: var(--admin-soft, #f8fafc);
  object-fit: contain;
  flex-shrink: 0;
  overflow: hidden;
}

.workshop-order-preview--sm {
  width: 24px;
  height: 24px;
}

.workshop-order-preview--md {
  width: 88px;
  height: 88px;
}

.workshop-order-preview--empty {
  color: var(--admin-text-muted, #94a3b8);
}

/* Секция внутри карточки заказа (Материалы / Фурнитура / Нанесение / Прочее).
   Каждая секция — компактный список строк. Левая граница рисует тон секции
   через CSS-переменные внутри модификатора `--material/--hardware/...` */
.workshop-need-section {
  display: flex;
  flex-direction: column;
  gap: 8px;
  border-left: 3px solid var(--workshop-need-section-tone, var(--admin-border, #e3e9f2));
  padding: 2px 0 2px 12px;
}

.workshop-need-section--material {
  --workshop-need-section-tone: #38bdf8;
  --workshop-need-section-text: #0369a1;
}

.workshop-need-section--hardware {
  --workshop-need-section-tone: #a78bfa;
  --workshop-need-section-text: #5b21b6;
}

.workshop-need-section--application {
  --workshop-need-section-tone: #f59e0b;
  --workshop-need-section-text: #92400e;
}

.workshop-need-section--other {
  --workshop-need-section-tone: #94a3b8;
  --workshop-need-section-text: #334155;
}

.workshop-need-section__header {
  display: flex;
  align-items: baseline;
  gap: 0.5rem;
}

.workshop-need-section__label {
  font-size: 0.72rem;
  font-weight: 700;
  letter-spacing: 0.06em;
  text-transform: uppercase;
  color: var(--workshop-need-section-text, var(--admin-text, #1e293b));
}

.workshop-need-section__count {
  font-size: 0.75rem;
  color: var(--admin-text-muted, #475569);
  font-weight: 600;
}

.workshop-need-section__rows {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 6px;
}

/* Backward-compat: старый класс `.workshop-need-section__list`
   (использовался ранее в lines-mode). Сохраняем для совместимости —
   там может ещё что-то рендериться через старый markup. */
.workshop-need-section__list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 6px;
}

/* Строка потребности внутри секции. Сетка адаптируется на узком экране —
   на широком стоит «описание | нужно | к закупке | поставщик | статус | open»;
   на узком schreibt подгрузить вертикально. */
.workshop-need-row {
  display: grid;
  grid-template-columns: minmax(0, 1.6fr) auto auto minmax(0, 1.1fr) auto auto;
  gap: 0.4rem 0.85rem;
  align-items: center;
  padding: 0.35rem 0.5rem;
  border-radius: 6px;
}

.workshop-need-row + .workshop-need-row {
  border-top: 1px solid var(--admin-border, #e2e8f0);
}

.workshop-need-row__main {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  min-width: 0;
}

.workshop-need-row__description {
  font-size: 0.88rem;
}

.workshop-need-row__note {
  font-size: 0.76rem;
}

.workshop-need-row__qty {
  display: flex;
  flex-direction: column;
  align-items: flex-end;
  font-size: 0.85rem;
}

.workshop-need-row__qty-label {
  font-size: 0.7rem;
  color: var(--admin-text-muted, #94a3b8);
}

.workshop-need-row__supplier {
  font-size: 0.85rem;
  min-width: 0;
}

.workshop-need-row__status {
  display: flex;
  align-items: center;
}

.workshop-need-row__open {
  white-space: nowrap;
}

.workshop-need-row__order-cell {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  min-width: 0;
}

.workshop-need-row__order-cell-text {
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  min-width: 0;
}

@media (max-width: 920px) {
  .workshop-need-row {
    grid-template-columns: minmax(0, 1fr) auto;
    grid-auto-flow: row;
  }

  .workshop-need-row__qty,
  .workshop-need-row__supplier,
  .workshop-need-row__status,
  .workshop-need-row__open {
    grid-column: span 2;
    align-items: flex-start;
  }

  .workshop-need-row__qty {
    flex-direction: row;
    gap: 0.4rem;
  }
}

/* Бейдж типа потребности (Material / Hardware / Application / Other).
   Тоны идут параллельно тонам левых границ секций — мягкий цвет фона
   плюс контрастный текст. */
.workshop-need-kind-badge {
  display: inline-flex;
  align-items: center;
  font-size: 0.72rem;
  font-weight: 600;
  letter-spacing: 0.02em;
  text-transform: uppercase;
  padding: 2px 8px;
  border-radius: 999px;
  border: 1px solid transparent;
  white-space: nowrap;
}

.workshop-need-kind-badge--material {
  color: #0369a1;
  background: #e0f2fe;
  border-color: #bae6fd;
}

.workshop-need-kind-badge--hardware {
  color: #5b21b6;
  background: #ede9fe;
  border-color: #ddd6fe;
}

.workshop-need-kind-badge--application {
  color: #92400e;
  background: #fef3c7;
  border-color: #fde68a;
}

.workshop-need-kind-badge--other {
  color: #334155;
  background: #f1f5f9;
  border-color: #e2e8f0;
}

/* ---------------------------------------------------------------------------
   /admin/workshop-needs — компактное inline-редактирование строк
   ---------------------------------------------------------------------------

   Polish-итерация «Себестоимость заказа» (см.
   `apps/web/app/admin/workshop-needs/inline-edit-row.tsx`).

   Раньше inline-форма редактирования рендерилась как двумерная
   `admin-form-grid`-табличка (по 2 колонки на каждый ряд) и
   занимала 4–5 экранов под каждую строку. Тратить место — нерационально:
   закупщик видит максимум одну строку на скролл и теряет «контекст
   соседних строк». Здесь форма перестроена в горизонтальный grid
   `.workshop-need-line` с фиксированными колонками — на desktop
   одна строка ≈ 56–72px, на узких экранах сетка сворачивается в
   вертикальную карточку через @media.

   Используется в двух местах с одним и тем же компонентом
   `<InlineEditWorkshopNeedRow>`:
     - `view=lines`  — полная строка с превью + клиентом + типом
                       (`showOrderInfo = true`, `bulkSelect = …`);
     - `view=orders` — компактная подстрока внутри `OrderNeedGroupCard`
                       (`showOrderInfo = false`, без bulk-PO).

   Сетка распределяет колонки через `data-cell="…"`-атрибуты, чтобы
   не плодить grid-template-areas — порядок ячеек = visual-порядок
   в DOM.
   --------------------------------------------------------------------------- */

.workshop-need-line {
  display: grid;
  grid-template-columns:
    minmax(220px, 1.4fr)        /* description */
    90px                        /* calc qty */
    100px                       /* purchase qty */
    110px                       /* unit price */
    90px                        /* currency */
    110px                       /* line total */
    minmax(140px, 1fr)          /* supplier text */
    120px                       /* expected date */
    minmax(150px, 0.8fr)        /* status (select + badge) */
    auto;                       /* save + detail link */
  gap: 6px 8px;
  align-items: end;
  padding: 0.5rem 0.6rem;
  border-radius: 8px;
  background: var(--admin-card, #ffffff);
  border: 1px solid var(--admin-border, #e2e8f0);
}

/* В режиме `view=lines` сверху ещё две колонки: «Заказ + клиент» и
   «Тип». Bulk-чекбокс приходит первым, если включён feature-flag
   `purchase-orders` (см. компонент). */
.workshop-need-line--with-order {
  /* 13 колонок — самый «тесный» layout. Минимумы намеренно сжаты,
     чтобы сумма колонок укладывалась в контент-область рабочих
     экранов (~1440px), а не уезжала за правый край, обрезая колонку
     «Сохранить»/«Подробнее». Если экран всё же ýже суммы — список
     `.workshop-need-line-list` даёт горизонтальный скролл, так что
     последняя колонка никогда не клипается насмерть (была причина
     «таблица не влезает в окно»). */
  grid-template-columns:
    auto                        /* bulk checkbox (optional) */
    minmax(150px, 1fr)          /* preview + order number + client */
    minmax(64px, auto)          /* kind badge */
    minmax(180px, 1.3fr)        /* description */
    74px                        /* calc qty */
    86px                        /* purchase qty */
    96px                        /* unit price */
    80px                        /* currency */
    96px                        /* line total */
    minmax(120px, 0.9fr)        /* supplier text */
    120px                       /* expected date */
    minmax(128px, 0.7fr)        /* status */
    auto;                       /* save + detail link */
}

.workshop-need-line + .workshop-need-line {
  margin-top: 6px;
}

/* Список строк в `view=lines` — простая вертикальная стопка
   inline-форм. Использует только padding + spacing; сама строка
   уже принимает grid-семантику (`.workshop-need-line`). */
.workshop-need-line-list {
  display: flex;
  flex-direction: column;
  gap: 6px;
  list-style: none;
  margin: 0;
  padding: 0;
  /* Container-query контекст: строки перестраиваются по ШИРИНЕ САМОГО
     СПИСКА, а не вьюпорта. Это и есть фикс «не влезает в окно»: вьюпорт
     не учитывает сайдбар админки (~280px), поэтому @media-брейкпоинт
     складывал строку слишком поздно и 13-колоночный грид обрезал
     колонку «Сохранить»/«Подробнее». `@container` ниже меряет реально
     доступную ширину. */
  container: wnlines / inline-size;
}

/* ---------------------------------------------------------------------------
   Этап «Цена продажи за единицу» — блок «Продажа» внутри
   `OrderCostEstimateCard` на /admin/orders/[id]. Read-only сводка
   цены/выручки/маржи; меняется из формы редактирования заказа.
   --------------------------------------------------------------------------- */
.admin-order-cost-estimate-card__sale {
  margin-top: 0.75rem;
  padding-top: 0.75rem;
  border-top: 1px dashed var(--admin-border, #e2e8f0);
}

.admin-order-cost-estimate-card__sale-title {
  margin: 0 0 0.4rem;
  font-size: 0.85rem;
  font-weight: 600;
  color: var(--admin-text, #1e293b);
}

.admin-order-cost-estimate-card__sale-list {
  margin: 0;
}

/* Колонка «Цена» в списке заказов (этап «Цена продажи за единицу»). */
.admin-order-price-cell {
  display: inline-flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 1px;
  white-space: nowrap;
}

.admin-order-price-cell__hint {
  font-size: 0.72rem;
}

.workshop-need-line__cell {
  display: flex;
  flex-direction: column;
  gap: 2px;
  min-width: 0;
}

.workshop-need-line__cell--check {
  justify-content: center;
  align-self: center;
  padding-top: 12px;
}

.workshop-need-line__cell--order {
  flex-direction: row;
  align-items: center;
  gap: 0.5rem;
}

.workshop-need-line__order-text {
  display: flex;
  flex-direction: column;
  gap: 1px;
  min-width: 0;
}

.workshop-need-line__order-number {
  font-size: 0.85rem;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}

.workshop-need-line__client {
  font-size: 0.72rem;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}

.workshop-need-line__cell--kind {
  align-self: center;
  align-items: flex-start;
  justify-content: center;
}

.workshop-need-line__cell--description {
  align-self: stretch;
  justify-content: center;
}

.workshop-need-line__description-text {
  font-size: 0.86rem;
  font-weight: 500;
  line-height: 1.25;
  word-break: break-word;
}

.workshop-need-line__note {
  font-size: 0.72rem;
  line-height: 1.25;
  display: -webkit-box;
  -webkit-box-orient: vertical;
  -webkit-line-clamp: 2;
  overflow: hidden;
}

/*
 * Этап «Исправить формирование Потребности цеха» (см. ТЗ §8 «UI
 * /admin/workshop-needs»). Под description строки потребности
 * рисуем secondary-блок с дополнительными характеристиками
 * фурнитуры (размер · материал · цвет) и preview изображения
 * материала. Если правило цвета `ORDER_SELECTED_COLOR` и цвет
 * не указан — рядом появляется warning-плашка.
 */
.workshop-need-line__description-row {
  display: flex;
  align-items: center;
  gap: 8px;
  min-width: 0;
}

.workshop-need-line__material-image {
  width: 32px;
  height: 32px;
  border-radius: 6px;
  object-fit: cover;
  background: var(--admin-bg-soft, #f1f5f9);
  flex: 0 0 auto;
}

.workshop-need-line__hardware-meta {
  font-size: 0.74rem;
  line-height: 1.25;
}

.workshop-need-line__color-warning {
  display: inline-block;
  margin-top: 2px;
  padding: 2px 8px;
  font-size: 0.72rem;
  line-height: 1.3;
  border-radius: 999px;
  background: var(--admin-tone-warning-bg, #fef3c7);
  color: var(--admin-tone-warning-fg, #92400e);
}

.workshop-need-line__cell--calc {
  align-items: flex-end;
  align-self: stretch;
  justify-content: center;
}

.workshop-need-line__qty-value {
  font-size: 0.85rem;
  white-space: nowrap;
}

.workshop-need-line__hint {
  font-size: 0.68rem;
  color: var(--admin-text-muted, #94a3b8);
  text-transform: uppercase;
  letter-spacing: 0.02em;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}

.workshop-need-line__cell--total {
  align-items: flex-end;
  align-self: stretch;
  justify-content: center;
}

.workshop-need-line__total {
  display: inline-flex;
  align-items: baseline;
  gap: 1px;
  white-space: nowrap;
  font-size: 0.9rem;
}

.workshop-need-line__total-currency {
  font-size: 0.8rem;
  color: var(--admin-text-muted, #94a3b8);
}

.workshop-need-line__cell--status {
  align-self: stretch;
  justify-content: flex-end;
  gap: 4px;
}

.workshop-need-line__cell--save {
  align-items: stretch;
  align-self: stretch;
  justify-content: flex-end;
  gap: 4px;
  flex-direction: column;
}

.workshop-need-line__detail-link {
  font-size: 0.72rem;
  text-align: center;
  white-space: nowrap;
}

/* Сжимаем стандартные input/select внутри grid-строки, чтобы они
   не «сожрали» соседние ячейки. */
.workshop-need-line input[type="text"],
.workshop-need-line input[type="date"],
.workshop-need-line select {
  width: 100%;
  font-size: 0.85rem;
  padding: 4px 6px;
  min-height: 30px;
  box-sizing: border-box;
}

.workshop-need-line .admin-btn {
  font-size: 0.78rem;
  padding: 5px 8px;
  white-space: nowrap;
}

.workshop-need-inline__alert {
  grid-column: 1 / -1;
  margin-top: 6px;
  display: flex;
  align-items: center;
  gap: 0.4rem;
  font-size: 0.78rem;
}

/* Комментарий закупщика — отдельная строка под grid-ом, на всю
   ширину. На широком экране grid имеет 13 колонок, поэтому
   проще выкатывать эту ячейку через `grid-column: 1 / -1`. */
.workshop-need-inline__comment {
  grid-column: 1 / -1;
  margin-top: 4px;
}

/* Когда ширина СПИСКА (не вьюпорта) меньше, чем нужно горизонтальной
   13-колоночной строке (~1430px), сворачиваемся в аккуратную карточку.
   Пороги заданы с запасом над реальной суммой колонок, чтобы строка
   никогда не клипалась за краем контент-области.

   Тиры (по ширине контейнера `.workshop-need-line-list`):
     < 1480px — карточка 4 колонки (плотно, но всё видно);
     < 1040px — 2 колонки (планшет);
     <  680px — 1 колонка (телефон).
   Order / Description / Save / Comment всегда на всю ширину. */
@container wnlines (max-width: 1480px) {
  .workshop-need-line,
  .workshop-need-line--with-order {
    grid-template-columns: repeat(4, minmax(0, 1fr));
    grid-auto-flow: row dense;
    align-items: end;
  }

  .workshop-need-line__cell--order,
  .workshop-need-line__cell--description,
  .workshop-need-line__cell--save {
    grid-column: 1 / -1;
  }

  .workshop-need-line__cell--save {
    flex-direction: row;
    flex-wrap: wrap;
    justify-content: flex-end;
    align-items: center;
  }
}

@container wnlines (max-width: 1040px) {
  .workshop-need-line,
  .workshop-need-line--with-order {
    grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  }
}

@container wnlines (max-width: 680px) {
  .workshop-need-line,
  .workshop-need-line--with-order {
    grid-template-columns: minmax(0, 1fr);
  }

  .workshop-need-line__cell {
    grid-column: 1 / -1;
  }
}

/* Comment-toggle: маленькая ghost-кнопка в правом краю строки.
   Если комментарий уже сохранён — добавляется маленький dot-бейдж
   справа от label, чтобы закупщик видел, что коммент есть. */
.workshop-order-need-row__comment-toggle {
  display: flex;
  align-items: center;
  justify-content: flex-end;
}

.workshop-order-need-row__comment-toggle--lines {
  grid-column: 1 / -1;
  justify-content: flex-start;
  margin-top: 4px;
}

.workshop-order-need-row__comment-button {
  font-size: 0.75rem;
  padding: 4px 8px;
  min-height: 28px;
  display: inline-flex;
  align-items: center;
  gap: 4px;
  white-space: nowrap;
  border-radius: 6px;
}

.workshop-order-need-row__comment-button--has {
  color: var(--admin-accent, #2563eb);
  border-color: var(--admin-accent-soft, #bfdbfe);
}

.workshop-order-need-row__comment-button--open {
  background: var(--admin-soft, #eff6ff);
}

.workshop-order-need-row__comment-button-label {
  font-size: 0.72rem;
}

.workshop-order-need-row__comment-dot {
  width: 6px;
  height: 6px;
  border-radius: 50%;
  background: var(--admin-accent, #2563eb);
  display: inline-block;
}

/* Раскрытый comment-блок — строка под grid-ом, textarea во всю
   ширину строки. */
.workshop-order-need-row__comment {
  grid-column: 1 / -1;
  margin-top: 4px;
}

.workshop-order-need-row__comment textarea {
  width: 100%;
  font-size: 0.85rem;
  padding: 6px 8px;
  border-radius: 6px;
  border: 1px solid var(--admin-border, #e3e9f2);
  background: #fff;
  resize: vertical;
  min-height: 56px;
  box-sizing: border-box;
}

.workshop-order-need-row__comment textarea:focus {
  outline: 2px solid var(--admin-accent, #2563eb);
  outline-offset: -1px;
  border-color: var(--admin-accent, #2563eb);
}

/* Свёрнутый read-only комментарий: текст под строкой с
   кликабельными ссылками (см. `renderCommentWithLinks`). */
.workshop-order-need-row__comment-text {
  margin: 0;
  font-size: 0.85rem;
  line-height: 1.4;
  color: var(--admin-text-muted, #5b6472);
  white-space: pre-wrap;
  word-break: break-word;
}

.workshop-order-need-row__comment-link {
  color: var(--admin-accent, #2563eb);
  text-decoration: underline;
  word-break: break-all;
}

.workshop-order-need-row__comment-link:hover {
  text-decoration: none;
}

/* Compact-вариант формы «Завершить расчёт» в header карточки заказа.
   Render: одна строка label + input + кнопка. Без больших
   `admin-form-grid`-блоков. */
.workshop-need-complete-form {
  display: flex;
  flex-direction: column;
  gap: 6px;
}

.workshop-need-complete-form__field {
  display: flex;
  flex-direction: column;
  gap: 2px;
}

.workshop-need-complete-form__field input {
  min-height: 32px;
  padding: 4px 8px;
  font-size: 0.85rem;
  border-radius: 6px;
  border: 1px solid var(--admin-border, #e3e9f2);
}

.workshop-need-complete-form__actions {
  display: flex;
  align-items: center;
  gap: 6px;
}

.workshop-need-complete-form__alert {
  font-size: 0.78rem;
  padding: 4px 8px;
}

.workshop-need-complete-form--compact {
  flex-direction: row;
  flex-wrap: wrap;
  align-items: center;
  gap: 8px;
}

.workshop-need-complete-form--compact .workshop-need-complete-form__field {
  flex-direction: row;
  align-items: center;
  gap: 6px;
}

.workshop-need-complete-form--compact .workshop-need-complete-form__field label {
  font-size: 0.75rem;
  color: var(--admin-text-muted, #475569);
  white-space: nowrap;
}

.workshop-need-complete-form--compact .workshop-need-complete-form__field input {
  width: 92px;
  font-size: 0.83rem;
  min-height: 32px;
  padding: 4px 8px;
}

.workshop-need-complete-form--compact .workshop-need-complete-form__actions {
  margin-left: 0;
}

.workshop-need-complete-form--compact .workshop-need-complete-form__alert {
  flex-basis: 100%;
}

/* Карточка заказа — ancestor контейнера `wnorows`, поэтому её
   адаптив остаётся на вьюпорт-`@media` (container query не умеет
   стилизовать предка своего контейнера). */
@media (max-width: 720px) {
  .workshop-order-group-card {
    padding: 14px;
  }

  .workshop-order-group-card__header {
    flex-direction: column;
    align-items: stretch;
  }

  .workshop-order-group-card__actions {
    width: 100%;
    justify-content: flex-start;
  }
}

/* ===========================================================================
   /admin/workshop-needs?view=orders — ЗОНАЛЬНАЯ строка потребности
   (`.wn-zrow`). Заменяет прежнюю 10-колоночную ленту, которая на
   реальной ширине (контент-область минус сайдбар < 1340px) схлопывалась
   в 4-колоночную «кашу» `auto-flow dense` без видимой группировки и с
   разновысокими полями.
   ---------------------------------------------------------------------------
   Поля неоднородны по смыслу, поэтому строка собрана из трёх зон-блоков
   с подписями и подложкой:
     • РАСЧЁТ    (read-only вывод системы): описание + «Нужно N ед.»;
     • ЗАКУПКА   (ввод закупщика + итог):   К закупке · Цена · Валюта → Сумма;
     • ЛОГИСТИКА (процесс):                 Поставщик · Поставка · Статус · ✔.
   Зоны — flex-айтемы: на широком layout идут в ряд, на узком переносятся
   в стек целиком (группировка сохраняется на любой ширине). Все контролы
   фиксированной высоты `--wn-fh` → поля в зоне выровнены.
   Используется только в orders-режиме (`showOrderInfo=false`); построчный
   вид `view=lines` остаётся на прежней `.workshop-need-line`.
   =========================================================================== */

.wn-zrow {
  --wn-fh: 34px;
  display: flex;
  flex-wrap: wrap;
  gap: 12px;
  align-items: stretch;
  background: #fff;
  border: 1px solid var(--admin-border, #e3e9f2);
  border-radius: 12px;
  padding: 10px 12px;
}

.wn-zrow + .wn-zrow {
  margin-top: 8px;
}

.wn-zone {
  flex: 1 1 300px;
  min-width: 0;
  display: flex;
  flex-direction: column;
  gap: 8px;
  border-radius: 10px;
  padding: 8px 10px;
}

.wn-zone--calc {
  flex-grow: 1.25;
  background: var(--admin-soft, #f8fafc);
}

.wn-zone--buy {
  flex-grow: 1.35;
  background: #f6f9ff;
}

.wn-zone--log {
  flex-grow: 1.15;
  background: #fafafa;
}

.wn-zone__cap {
  font-size: 0.62rem;
  font-weight: 700;
  letter-spacing: 0.07em;
  text-transform: uppercase;
  color: var(--admin-muted, #64748b);
  padding-bottom: 5px;
  border-bottom: 2px solid var(--wn-cap-tone, var(--admin-border, #e2e8f0));
}

.wn-zone--buy .wn-zone__cap {
  color: var(--admin-accent, #2563eb);
  --wn-cap-tone: #93c5fd;
}

.wn-zone__body {
  flex: 1 1;
  min-width: 0;
}

/* --- общие поля внутри зон --- */
.wn-field {
  display: flex;
  flex-direction: column;
  gap: 3px;
  min-width: 0;
}

.wn-field__lab {
  font-size: 0.62rem;
  font-weight: 600;
  letter-spacing: 0.02em;
  text-transform: uppercase;
  color: var(--admin-text-muted, #94a3b8);
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}

.wn-zrow input[type='text'],
.wn-zrow input[type='date'],
.wn-zrow select {
  width: 100%;
  box-sizing: border-box;
  height: var(--wn-fh);
  font-size: 0.85rem;
  padding: 4px 8px;
  border-radius: 7px;
  border: 1px solid var(--admin-border, #dbe2ea);
  background: #fff;
}

.wn-zrow input:focus,
.wn-zrow select:focus {
  outline: 2px solid var(--admin-accent, #2563eb);
  outline-offset: -1px;
  border-color: var(--admin-accent, #2563eb);
}

.wn-zrow input:disabled,
.wn-zrow select:disabled {
  background: var(--admin-soft, #f1f5f9);
  color: var(--admin-muted, #94a3b8);
}

/* read-only вывод расчёта — «чип» той же высоты, визуально не редактируемый */
.wn-readout {
  align-items: stretch;
}

.wn-readout__box {
  height: var(--wn-fh);
  display: flex;
  align-items: center;
  justify-content: flex-end;
  width: 100%;
  box-sizing: border-box;
  padding: 0 10px;
  border-radius: 7px;
  border: 1px dashed #cbd5e1;
  background: #eef2f7;
  font-size: 0.9rem;
  font-weight: 700;
  color: var(--admin-text, #1e293b);
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}

.wn-readout--sum .wn-readout__box {
  border-style: solid;
  border-color: #bcd3f5;
  background: #eaf2ff;
  color: var(--admin-accent, #1d4ed8);
}

/* --- РАСЧЁТ: описание + «Нужно» --- */
.wn-zone__body--calc {
  display: flex;
  gap: 10px;
  align-items: flex-start;
}

.wn-desc {
  flex: 1 1;
  min-width: 0;
  display: flex;
  flex-direction: column;
  gap: 3px;
}

.wn-desc__row {
  display: flex;
  align-items: center;
  gap: 8px;
  min-width: 0;
}

.wn-desc__img {
  width: 30px;
  height: 30px;
  border-radius: 6px;
  object-fit: cover;
  flex: 0 0 auto;
  background: var(--admin-soft, #f1f5f9);
}

.wn-desc__text {
  font-size: 0.92rem;
  font-weight: 600;
  line-height: 1.3;
  word-break: break-word;
}

.wn-desc__meta {
  font-size: 0.74rem;
  color: var(--admin-muted, #64748b);
  line-height: 1.25;
}

.wn-desc__warning {
  align-self: flex-start;
  margin-top: 2px;
  padding: 2px 8px;
  font-size: 0.72rem;
  line-height: 1.3;
  border-radius: 999px;
  background: var(--admin-tone-warning-bg, #fef3c7);
  color: var(--admin-tone-warning-fg, #92400e);
}

.wn-zone__body--calc .wn-readout {
  flex: 0 0 92px;
}

/* --- ЗАКУПКА: К закупке · Цена · Валюта, ниже = Сумма --- */
.wn-zone__body--buy {
  display: flex;
  flex-wrap: wrap;
  gap: 8px;
}

.wn-zone__body--buy > .wn-field {
  flex: 1 1 78px;
}

.wn-zone__body--buy > .wn-readout--sum {
  flex: 1 1 100%;
}

/* --- ЛОГИСТИКА: Поставщик (вся ширина) · Поставка · Статус · ✔ --- */
.wn-zone__body--log {
  display: flex;
  flex-wrap: wrap;
  gap: 8px;
  align-items: flex-end;
}

.wn-zone__body--log > .wn-field--supplier {
  flex: 1 1 100%;
}

.wn-zone__body--log > .wn-field--supplier select {
  margin-bottom: 4px;
}

.wn-zone__body--log > .wn-field--date {
  flex: 1 1 120px;
}

.wn-zone__body--log > .wn-field--status {
  flex: 2 1 130px;
}

.wn-zone__body--log > .wn-field--save {
  flex: 0 0 auto;
  align-self: flex-end;
}

.wn-zone__body--log .wn-save {
  height: var(--wn-fh);
  min-width: 42px;
  border: 0;
  border-radius: 8px;
  background: var(--admin-accent, #2563eb);
  color: #fff;
  cursor: pointer;
  display: inline-flex;
  align-items: center;
  justify-content: center;
}

.wn-zone__body--log .wn-save:disabled {
  opacity: 0.55;
  cursor: default;
}

/* --- подвал строки: comment-toggle + раскрытый коммент + алерты --- */
.wn-zrow__foot {
  flex: 1 1 100%;
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 8px;
  margin-top: 2px;
}

/* Ссылка «Подробности» → полная карточка /admin/workshop-needs/[id].
   Прижата вправо подвала строки, отдельно от comment-toggle слева. */
.wn-zrow__detail-link {
  margin-left: auto;
  font-size: 0.78rem;
  white-space: nowrap;
}

.wn-zrow__comment {
  flex: 1 1 100%;
}

.wn-zrow__comment textarea {
  width: 100%;
  box-sizing: border-box;
  font-size: 0.85rem;
  padding: 6px 8px;
  border-radius: 7px;
  border: 1px solid var(--admin-border, #e3e9f2);
  background: #fff;
  resize: vertical;
  min-height: 52px;
}

.wn-zrow__comment-view {
  flex: 1 1 100%;
}

.wn-zrow .wn-zrow__alert {
  flex: 1 1 100%;
  display: flex;
  align-items: center;
  gap: 0.4rem;
  font-size: 0.78rem;
}

/* ============================================================================
   Карточка операции — компактный блок «Экономика операции» (BY_SIZE).
   ----------------------------------------------------------------------------
   Раньше для `BY_SIZE` рисовалась широкая `.admin-table` с пятью
   колонками и одной строкой на размер; в правой колонке
   `admin-grid-2` (минимальная по дизайну) она не помещалась и часто
   содержала 21 одинаковую строку. Теперь helper
   `groupOperationEconomicsRows` (`apps/web/lib/operation-economics.ts`)
   схлопывает одинаковые строки в группы, а UI рендерит компактные
   карточки-группы внутри `.operation-economics-groups`.

   Цели вёрстки:
     - помещается в правую колонку и не вызывает horizontal overflow;
     - метрики (Ставка / Норма / 8ч / Доход) переносятся внутри
       карточки на узких экранах;
     - визуально согласуется с остальной admin-палитрой
       (`--admin-border` / `--admin-soft` / `--admin-muted`).
   ============================================================================ */

.operation-economics-groups {
  display: grid;
  gap: 8px;
}

.operation-economics-group {
  border: 1px solid var(--admin-border);
  border-radius: 12px;
  padding: 10px 12px;
  background: var(--admin-soft);
  min-width: 0;
}

.operation-economics-group__sizes {
  font-weight: 700;
  margin-bottom: 8px;
  color: var(--admin-text);
  word-break: break-word;
}

.operation-economics-group__metrics {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: 6px 10px;
}

.operation-economics-group__metric {
  display: flex;
  flex-direction: column;
  min-width: 0;
}

.operation-economics-group__label {
  display: block;
  font-size: 11px;
  color: var(--admin-muted);
  text-transform: uppercase;
  letter-spacing: 0.02em;
  line-height: 1.2;
}

.operation-economics-group__value {
  font-weight: 700;
  color: var(--admin-text);
  word-break: break-word;
}

@media (max-width: 1199px) {
  .operation-economics-group__metrics {
    grid-template-columns: 1fr;
  }
}

/* ============================================================================
   Категории номенклатуры — компактная колонка «Единица потребности».
   ----------------------------------------------------------------------------
   Колонка `unit` в таблице параметров категории
   (`apps/web/app/admin/pattern-categories/new/create-pattern-category-form.tsx`,
   `apps/web/app/admin/pattern-categories/[id]/edit-pattern-category-form.tsx`)
   раньше растягивалась длинным муту-текстом «В какой единице строка
   попадёт в Потребность цеха». Это раздувало строку и плодило 2 строки
   высоты на каждый параметр. Теперь:
     - ширина колонки задаётся через `<colgroup>` (auto layout уважает
       width на `<col>`, когда контент в ячейке короткий — у нас там
       только select);
     - пояснение переезжает в `title`-атрибут на label-обёртке
       (`.admin-inline-help`) с маленьким значком ⓘ;
     - select unit-а ужимается до 150 px (`max-width`, `min-width: 0`),
       чтобы не «выпрыгивать» за пределы ячейки.

   `table-layout: auto` оставлен сознательно: в строке есть текстовый
   `<input>` с placeholder и select-ы, у которых ширина важна для
   читаемости; жёсткий fixed-режим обрезает их на средних экранах.
   ============================================================================ */

.pattern-category-param-table__unit-col {
  width: 150px;
}

.pattern-category-param-table__required-col {
  width: 110px;
}

.pattern-category-param-table__actions-col {
  width: 56px;
}

.pattern-category-param-row__unit {
  width: 100%;
  min-width: 0;
  max-width: 150px;
  box-sizing: border-box;
}

.admin-inline-help {
  cursor: help;
  border-bottom: 1px dotted currentColor;
}

.admin-inline-help__icon {
  display: inline-block;
  margin-left: 0.2rem;
  color: var(--admin-muted);
  font-size: 0.85em;
  cursor: help;
}

@media (max-width: 720px) {
  /* Внутри admin-table карточек на узком экране (см. media-query выше
     `@media (max-width: 720px) .admin-table`) ограничение в 150 px
     мешает: select растягивается на всю карточку. */
  .pattern-category-param-row__unit {
    max-width: none;
  }
}

/* =============================================================================
   Order workspace — единый layout мастера/карточки заказа
   ----------------------------------------------------------------------------
   Цель — сделать так, чтобы `/admin/orders/new`, `/admin/orders/[id]` и
   `/admin/orders/[id]/edit` визуально были одной и той же страницей с
   разными состояниями. Layout строит `OrderWorkspaceLayout`, hero —
   `OrderHeroCard`, ряд вкладок — `OrderDetailTabs`. Empty-state в
   create-mode — `OrderTabEmptyState`.

   Источник правды по списку вкладок и их лейблам — `ORDER_DETAIL_TABS`
   (см. `apps/web/components/orders/order-detail-tabs-config.ts`).
   Backend / Prisma не трогаем — это presentation-слой.
   ============================================================================= */

.order-workspace {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
  width: 100%;
  max-width: 100%;
  min-width: 0;
  box-sizing: border-box;
}

.order-workspace__hero {
  display: contents;
}

.order-workspace__tabs {
  width: 100%;
  min-width: 0;
}

.order-workspace__body {
  width: 100%;
  min-width: 0;
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
}

/* --- Hero card («Основное») ---------------------------------------------- */
/*
  Order workspace v2: hero — это рабочий блок управления заказом, а не
  «инфо-простыня». Он содержит:
    1. Header (eyebrow + title + status + createdAt);
    2. короткое product-summary (ОДНА строка, без превью);
    3. форма «Основное» (`.order-hero-card__basic-form`) — управленческие
       поля (подразделение / срок / клиент / цена / комментарий);
    4. KPI-чипы (`.order-hero-card__kpis`);
    5. workflow actions (`.order-hero-card__actions`).
*/

.order-hero-card {
  display: flex;
  flex-direction: column;
  gap: 0.85rem;
  background: var(--admin-surface);
  border: 1px solid #dbe3ee;
  border-radius: 20px;
  padding: 1.1rem 1.25rem;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04),
    0 8px 24px rgba(15, 23, 42, 0.05);
  width: 100%;
  max-width: 100%;
  min-width: 0;
  box-sizing: border-box;
}

.order-hero-card--create {
  border-color: #d8e1ee;
  background: linear-gradient(
    135deg,
    var(--admin-surface) 60%,
    rgba(99, 102, 241, 0.05) 100%
  );
}

.order-hero-card--view {
  border-color: #dbe3ee;
}

.order-hero-card--edit {
  border-color: #fde68a;
  background: linear-gradient(
    135deg,
    var(--admin-surface) 60%,
    rgba(250, 204, 21, 0.05) 100%
  );
}

.order-hero-card__alerts {
  display: flex;
  flex-wrap: wrap;
  gap: 0.5rem;
}

.order-hero-card__head {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 0.75rem;
  flex-wrap: wrap;
}

.order-hero-card__identity {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  min-width: 0;
}

.order-hero-card__eyebrow {
  font-size: 0.72rem;
  font-weight: 600;
  letter-spacing: 0.06em;
  text-transform: uppercase;
  color: var(--admin-muted);
}

.order-hero-card__title {
  margin: 0;
  font-size: 1.4rem;
  font-weight: 700;
  letter-spacing: -0.01em;
  color: var(--admin-text);
  line-height: 1.2;
  word-break: break-word;
}

.order-hero-card__createdAt {
  font-size: 0.78rem;
  color: var(--admin-muted);
}

.order-hero-card__status {
  display: flex;
  align-items: center;
  gap: 0.5rem;
}

/* Короткая secondary-строка про продукцию — НЕ дублирует Product tab. */
.order-hero-card__product-summary {
  font-size: 0.85rem;
  color: var(--admin-text);
  padding: 0.4rem 0.6rem;
  background: rgba(99, 102, 241, 0.06);
  border-radius: 8px;
  border: 1px solid rgba(99, 102, 241, 0.12);
  word-break: break-word;
}

.order-hero-card__product-summary-text {
  display: inline;
}

.order-hero-card__product-summary-label {
  font-weight: 600;
  color: var(--admin-muted);
  text-transform: uppercase;
  font-size: 0.7rem;
  letter-spacing: 0.04em;
  margin-right: 0.25rem;
}

.order-hero-card__product-summary-meta {
  color: var(--admin-muted);
}

/* Форма «Основное». В create-mode это часть общей <form>, в view-mode —
   отдельная inline-form через updateOrderBasicsAction. */
.order-hero-card__basic-form {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
}

.order-hero-card__basic-form-inner {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
}

.order-hero-card__basic-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
  gap: 0.6rem 0.85rem;
}

.order-hero-card__field {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  min-width: 0;
}

.order-hero-card__field--comment {
  grid-column: 1 / -1;
}

.order-hero-card__field--price {
  min-width: 220px;
}

.order-hero-card__field label {
  font-size: 0.72rem;
  font-weight: 600;
  letter-spacing: 0.03em;
  color: var(--admin-muted);
  text-transform: uppercase;
}

/* Строка подписи поля: лейбл + иконка-инфо. Фиксированная высота
   (≈ 2 строки лейбла) держит все инпуты строки на одной линии вне
   зависимости от того, переносится подпись на 2 строки или нет. */
.order-hero-card__field-head {
  display: flex;
  align-items: flex-start;
  gap: 0.3rem;
  min-height: 2rem;
}

.order-hero-card__field-head label {
  flex: 1 1 auto;
  min-width: 0;
  line-height: 1.25;
}

.order-hero-card__field input,
.order-hero-card__field select,
.order-hero-card__field textarea {
  font: inherit;
  padding: 0.4rem 0.55rem;
  border: 1px solid var(--admin-border);
  border-radius: 8px;
  background: var(--admin-surface);
  color: var(--admin-text);
  width: 100%;
  box-sizing: border-box;
}

.order-hero-card__field textarea {
  min-height: 2.4rem;
  resize: vertical;
}

.order-hero-card__field input:disabled,
.order-hero-card__field select:disabled,
.order-hero-card__field textarea:disabled {
  background: rgba(0, 0, 0, 0.03);
  color: var(--admin-muted);
  cursor: not-allowed;
}

.order-hero-card__field-hint {
  font-size: 0.75rem;
  color: var(--admin-muted);
}

.order-hero-card__field-error {
  font-size: 0.75rem;
  color: var(--admin-danger-fg, #b91c1c);
}

/* Иконка-инфо рядом с подписью поля + тултип (заменяет текстовые
   подсказки под инпутом). */
.order-hero-card__field-info {
  position: relative;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  flex: 0 0 auto;
  width: 1.05rem;
  height: 1.05rem;
  margin-top: -0.05rem;
  border-radius: 50%;
  color: var(--admin-muted);
  cursor: help;
  outline: none;
}

.order-hero-card__field-info:hover,
.order-hero-card__field-info:focus-visible {
  color: var(--admin-accent, #2563eb);
}

.order-hero-card__field-info:focus-visible {
  box-shadow: 0 0 0 2px var(--admin-accent-soft, rgba(37, 99, 235, 0.35));
}

.order-hero-card__field-tip {
  position: absolute;
  bottom: calc(100% + 0.4rem);
  left: 50%;
  transform: translateX(-50%) translateY(0.2rem);
  z-index: 30;
  width: max-content;
  max-width: 240px;
  padding: 0.45rem 0.6rem;
  border-radius: 8px;
  background: var(--admin-text, #0f172a);
  color: #fff;
  font-size: 0.75rem;
  font-weight: 400;
  line-height: 1.35;
  text-transform: none;
  letter-spacing: normal;
  white-space: normal;
  box-shadow: 0 8px 24px rgba(15, 23, 42, 0.25);
  opacity: 0;
  visibility: hidden;
  pointer-events: none;
  transition: opacity 120ms ease, transform 120ms ease;
}

.order-hero-card__field-tip::after {
  content: '';
  position: absolute;
  top: 100%;
  left: 50%;
  transform: translateX(-50%);
  border: 5px solid transparent;
  border-top-color: var(--admin-text, #0f172a);
}

.order-hero-card__field-info:hover .order-hero-card__field-tip,
.order-hero-card__field-info:focus .order-hero-card__field-tip,
.order-hero-card__field-info:focus-visible .order-hero-card__field-tip {
  opacity: 1;
  visibility: visible;
  transform: translateX(-50%) translateY(0);
}

.order-hero-card__price-row {
  display: grid;
  grid-template-columns: 1fr 90px;
  gap: 0.4rem;
}

.order-hero-card__basic-form-actions {
  display: flex;
  align-items: center;
  flex-wrap: wrap;
  gap: 0.6rem;
}

.order-hero-card__basic-form-toast {
  font-size: 0.78rem;
  color: var(--admin-success-fg, #166534);
  background: var(--admin-success-bg, #dcfce7);
  border: 1px solid var(--admin-success-border, #bbf7d0);
  padding: 0.2rem 0.5rem;
  border-radius: 8px;
}

/* KPI chips */
.order-hero-card__kpis {
  list-style: none;
  margin: 0;
  padding: 0.5rem 0 0;
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(140px, 1fr));
  gap: 0.5rem;
  border-top: 1px dashed var(--admin-border);
}

.order-hero-card__kpi {
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  min-width: 0;
  padding: 0.4rem 0.55rem;
  border-radius: 8px;
  background: rgba(15, 23, 42, 0.03);
  border: 1px solid transparent;
}

.order-hero-card__kpi--success {
  background: var(--admin-success-bg, #dcfce7);
  border-color: var(--admin-success-border, #bbf7d0);
}

.order-hero-card__kpi--warning {
  background: var(--admin-warning-bg, #fef3c7);
  border-color: var(--admin-warning-border, #fde68a);
}

.order-hero-card__kpi--danger {
  background: var(--admin-danger-bg, #fee2e2);
  border-color: var(--admin-danger-border, #fecaca);
}

.order-hero-card__kpi--info {
  background: var(--admin-info-bg, #dbeafe);
  border-color: var(--admin-info-border, #bfdbfe);
}

.order-hero-card__kpi-label {
  font-size: 0.7rem;
  font-weight: 600;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--admin-muted);
}

.order-hero-card__kpi-value {
  font-size: 0.95rem;
  font-weight: 600;
  color: var(--admin-text);
  word-break: break-word;
}

.order-hero-card__kpi-unit {
  font-size: 0.78rem;
  color: var(--admin-muted);
  font-weight: 500;
}

/* Workflow actions row */
.order-hero-card__actions {
  display: flex;
  flex-wrap: wrap;
  gap: 0.5rem;
  align-items: center;
  justify-content: flex-start;
  min-width: 0;
  padding-top: 0.5rem;
  border-top: 1px dashed var(--admin-border);
}

.order-hero-card__actions .admin-btn {
  white-space: nowrap;
}

@media (max-width: 1100px) {
  .order-hero-card {
    padding: 1rem 1rem;
  }
}

@media (max-width: 640px) {
  .order-hero-card {
    padding: 0.85rem 0.85rem;
  }
  .order-hero-card__basic-grid {
    grid-template-columns: minmax(0, 1fr);
  }
  .order-hero-card__kpis {
    grid-template-columns: repeat(2, minmax(0, 1fr));
  }
}

/* --- Tabs ---------------------------------------------------------------- */

.order-detail-tabs {
  display: flex;
  flex-wrap: wrap;
  gap: 0.25rem;
  padding: 0.35rem;
  background: var(--admin-soft, #f1f5f9);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  width: 100%;
  min-width: 0;
  box-sizing: border-box;
}

.order-detail-tabs__link {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.5rem 0.85rem;
  border-radius: 10px;
  font-size: 0.88rem;
  font-weight: 600;
  color: var(--admin-text);
  background: transparent;
  border: 1px solid transparent;
  text-decoration: none;
  cursor: pointer;
  transition: background 140ms ease, color 140ms ease,
    border-color 140ms ease;
  white-space: nowrap;
}

.order-detail-tabs__link:hover:not(.order-detail-tabs__link--disabled):not(.order-detail-tabs__link--active) {
  background: rgba(255, 255, 255, 0.7);
  color: var(--admin-text);
}

.order-detail-tabs__link--active {
  background: var(--admin-surface);
  color: var(--admin-text);
  border-color: #c7d2fe;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04),
    0 6px 16px rgba(15, 23, 42, 0.05);
}

.order-detail-tabs__link--disabled {
  color: var(--admin-muted);
  cursor: not-allowed;
  opacity: 0.7;
}

.order-detail-tabs__icon {
  opacity: 0.7;
}

.order-detail-tabs__label {
  display: inline-block;
}

@media (max-width: 720px) {
  .order-detail-tabs {
    overflow-x: auto;
    flex-wrap: nowrap;
  }
}

/* --- Empty state for tabs without data ---------------------------------- */

.order-tab-empty-state {
  display: flex;
  flex-direction: column;
  align-items: flex-start;
  gap: 0.45rem;
  padding: 1.5rem 1.25rem;
  background: var(--admin-surface);
  border: 1px dashed var(--admin-border);
  border-radius: 16px;
  color: var(--admin-text);
  width: 100%;
  min-width: 0;
  box-sizing: border-box;
}

.order-tab-empty-state__title {
  font-size: 0.95rem;
  font-weight: 650;
  color: var(--admin-text);
}

.order-tab-empty-state__hint {
  margin: 0;
  font-size: 0.85rem;
  color: var(--admin-muted);
  line-height: 1.45;
}

.order-tab-empty-state__action {
  margin-top: 0.4rem;
}

/* --- Form / size matrix grid in workspace body --------------------------- */

.order-product-form-grid {
  display: grid;
  gap: var(--admin-space-md);
}

.order-size-matrix-card {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
}

/* --- Tab panels («Продукция», «Материалы», «Операции», …) --------------- */
/*
  Все tab-panel-ы пользуются одной и той же оболочкой `.order-tab-panel`,
  чтобы стиль (gap между карточками, ширина) не разъезжался между
  вкладками. Локальные модификаторы (`order-product-tab`, `order-materials-tab`
  и т.п.) дают возможность тонко настроить отступы внутри конкретной
  вкладки, не трогая общую обёртку.
*/

.order-tab-panel {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
  width: 100%;
  min-width: 0;
  box-sizing: border-box;
}

.order-tab-empty-states {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  margin-top: var(--admin-space-md);
}

/* Product tab — карточки лекала / превью / маршрут — две колонки на десктопе */
.order-product-tab {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
}

.order-product-tab__grid {
  display: grid;
  grid-template-columns: minmax(0, 2fr) minmax(0, 1fr);
  gap: var(--admin-space-md);
  align-items: start;
}

.order-product-tab__col-main,
.order-product-tab__col-aside {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
  min-width: 0;
}

@media (max-width: 1199px) {
  .order-product-tab__grid {
    grid-template-columns: minmax(0, 1fr);
  }
}

.order-materials-tab,
.order-operations-tab,
.order-logistics-tab,
.order-summary-tab,
.order-recommendations-tab {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
}

/* --- Recommendations card ------------------------------------------------ */

.order-recommendations-card {
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  padding: 0.9rem 1rem;
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
}

.order-recommendations-card--empty {
  flex-direction: row;
  align-items: center;
  gap: 0.7rem;
  background: var(--admin-success-bg, #dcfce7);
  border-color: var(--admin-success-border, #bbf7d0);
  color: var(--admin-success-fg, #166534);
}

.order-recommendations-card__head {
  display: flex;
  align-items: baseline;
  gap: 0.5rem;
  justify-content: space-between;
}

.order-recommendations-card__title {
  margin: 0;
  font-size: 1rem;
  font-weight: 700;
  color: var(--admin-text);
}

.order-recommendations-card__count {
  font-size: 0.78rem;
  font-weight: 600;
  color: var(--admin-muted);
  background: rgba(0, 0, 0, 0.04);
  padding: 0.1rem 0.45rem;
  border-radius: 999px;
}

.order-recommendations-card__list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
}

.order-recommendations-card__item {
  display: flex;
  gap: 0.55rem;
  align-items: flex-start;
  padding: 0.55rem 0.65rem;
  border-radius: 10px;
  background: rgba(15, 23, 42, 0.03);
  border-left: 3px solid var(--admin-border);
}

.order-recommendations-card__item--info {
  background: var(--admin-info-bg, #dbeafe);
  border-left-color: var(--admin-info-border, #93c5fd);
}

.order-recommendations-card__item--warning {
  background: var(--admin-warning-bg, #fef3c7);
  border-left-color: var(--admin-warning-border, #fcd34d);
}

.order-recommendations-card__item--danger {
  background: var(--admin-danger-bg, #fee2e2);
  border-left-color: var(--admin-danger-border, #fca5a5);
}

.order-recommendations-card__icon {
  flex: 0 0 auto;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 22px;
  height: 22px;
  color: var(--admin-muted);
}

.order-recommendations-card__icon--info {
  color: var(--admin-info-fg, #1e3a8a);
}

.order-recommendations-card__icon--warning {
  color: var(--admin-warning-fg, #9a3412);
}

.order-recommendations-card__icon--danger {
  color: var(--admin-danger-fg, #b91c1c);
}

.order-recommendations-card__item-body {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  min-width: 0;
}

.order-recommendations-card__item-title {
  font-size: 0.9rem;
  font-weight: 600;
  color: var(--admin-text);
}

.order-recommendations-card__item-hint {
  font-size: 0.8rem;
  color: var(--admin-muted);
}

/* ============================================================ */
/* Materials tab — unified table «Материалы»                    */
/* ============================================================ */
/*
  Объединённая таблица материалов (`OrderMaterialsUnifiedTable`) и
  компактный блок ручной разблокировки кроя
  (`ManualMaterialArrivalActions`) живут во вкладке «Материалы»
  карточки заказа `/admin/orders/[id]`. Раньше тут было три
  отдельных карточки («Потребность цеха» / «Себестоимость» /
  «Готовность к крою»), теперь — одна compact-таблица + одна
  кнопка снизу. Backend / Prisma / WorkshopNeed-формулы /
  CutReadinessService не менялись.
*/

.order-materials-table-card {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  padding: 0.7rem 0.9rem 0.85rem;
  min-width: 0;
}

.order-materials-table-card__head {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  flex-wrap: wrap;
}

.order-materials-table-card__title {
  font-size: 0.95rem;
  font-weight: 700;
  color: var(--admin-text);
  margin-right: 0.25rem;
}

.order-materials-table-card__summary {
  font-size: 0.78rem;
  color: var(--admin-muted);
  flex: 1 1 auto;
  min-width: 0;
}

.order-materials-table-card__summary-blockers {
  color: var(--admin-warning-fg, #9a3412);
  font-weight: 600;
}

.order-materials-table-card__edit-link {
  font-size: 0.8rem;
}

.order-materials-table-card__errors {
  margin: 0;
  padding: 0.4rem 0.6rem;
  list-style: none;
  background: var(--admin-warning-bg, #fef3c7);
  border: 1px solid var(--admin-warning-border, #fcd34d);
  color: var(--admin-warning-fg, #92400e);
  border-radius: 6px;
  font-size: 0.78rem;
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
}

.order-materials-table-card__errors li {
  margin: 0;
  padding: 0;
}

.order-materials-table-wrap {
  /* Внешняя обёртка — НЕ scroll-контейнер. Скролл живёт на внутреннем
     `.admin-table-wrap` (см. ниже), чтобы sticky-колонки прилипали к
     одному прокручиваемому предку. Здесь только не даём блоку вытолкнуть
     грид карточки. */
  width: 100%;
  max-width: 100%;
  min-width: 0;
  overflow: visible;
}

.order-materials-table {
  /* AdminTable монтирует <div class="admin-table-wrap"><table class="admin-table"/>.
     Этот div — единственный горизонтальный scroll-контейнер таблицы.
     База `.admin-table-wrap` имеет `overflow: hidden` (под border-radius);
     переопределяем X на `auto`, чтобы 1380px-таблица прокручивалась
     ВНУТРИ карточки, а не вылезала за неё и не обрезалась. min-width
     здесь НЕ ставим — ширину держит только таблица ниже, а div
     остаётся `max-width: 100%`. overflow-y оставляем hidden (из
     shorthand базы) — высоту это не режет (контент по высоте влезает),
     зато sticky-колонки прилипают именно к этому контейнеру. */
  font-size: 0.81rem;
  max-width: 100%;
  overflow-x: auto;
}

.order-materials-table-wrap .admin-table {
  /* 14 колонок (роль … комментарий) — таблице нужна минимальная ширина,
     чтобы поля не сжимались в кашу. Перебор ширины уезжает в
     горизонтальный скролл внутреннего `.admin-table-wrap`. */
  min-width: 1380px;
}

/* Sticky leading columns (Роль + Описание): при горизонтальном скролле
   контекст строки остаётся виден. Прилипают к scroll-контейнеру
   `.order-materials-table` (внутренний `.admin-table-wrap` с
   overflow-x: auto). Фиксируем ширину 1-й колонки, чтобы offset 2-й был
   стабилен. */
.order-materials-table .admin-table th:first-child,
.order-materials-table .admin-table td:first-child {
  position: sticky;
  left: 0;
  z-index: 1;
  width: 116px;
  min-width: 116px;
  max-width: 116px;
}
.order-materials-table .admin-table th:nth-child(2),
.order-materials-table .admin-table td:nth-child(2) {
  position: sticky;
  left: 116px;
  z-index: 1;
  /* правая граница-«тень» отделяет липкую зону от прокручиваемой */
  box-shadow: inset -1px 0 0 var(--admin-border);
}
/* Заголовки липких колонок — над липкими ячейками тела. */
.order-materials-table .admin-table thead th:first-child,
.order-materials-table .admin-table thead th:nth-child(2) {
  z-index: 3;
}
/* Непрозрачный фон, чтобы прокручиваемые ячейки не просвечивали.
   thead уже имеет `background: var(--admin-soft)`. */
.order-materials-table .admin-table tbody td:first-child,
.order-materials-table .admin-table tbody td:nth-child(2) {
  background: var(--admin-surface);
}
.order-materials-table .admin-table tbody tr:hover td:first-child,
.order-materials-table .admin-table tbody tr:hover td:nth-child(2) {
  background: var(--admin-soft);
}

.order-materials-table-wrap .admin-table th,
.order-materials-table-wrap .admin-table td {
  padding: 0.4rem 0.5rem;
  vertical-align: top;
}

.order-materials-table__role {
  font-weight: 600;
  color: var(--admin-text);
  /* В sticky-колонке фиксированной ширины (116px) длинные роли
     («Дополнительное полотно», «Дублерин / клеевые») переносим, а не
     обрезаем. */
  white-space: normal;
  word-break: break-word;
}

.order-materials-table__description {
  display: flex;
  align-items: flex-start;
  gap: 0.5rem;
  min-width: 0;
}

.order-materials-table__description-body {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  min-width: 0;
}

.order-materials-table__description-main {
  font-weight: 500;
  color: var(--admin-text);
  word-break: break-word;
}

.order-materials-table__description-meta {
  font-size: 0.74rem;
  color: var(--admin-muted);
  word-break: break-word;
}

.order-materials-table__image {
  flex: 0 0 auto;
  width: 36px;
  height: 36px;
  border-radius: 6px;
  background: var(--admin-bg, #f1f5f9);
  border: 1px solid var(--admin-border, #e2e8f0);
  overflow: hidden;
  display: inline-flex;
  align-items: center;
  justify-content: center;
}

.order-materials-table__image img {
  width: 100%;
  height: 100%;
  object-fit: cover;
}

.order-materials-table__image--empty {
  color: var(--admin-muted);
  background: transparent;
}

.order-materials-table__qty {
  display: inline-flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 0.1rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

.order-materials-table__qty--placeholder {
  color: var(--admin-muted);
}

.order-materials-table__qty-hint {
  font-size: 0.7rem;
  color: var(--admin-muted);
  font-weight: 400;
}

/* Frontend-итерация «план/факт» поверх MaterialIssue:
   компактный блок «План / Факт / Δ» внутри одной ячейки.
   Каждая строка — `label` (узкий) + `value` (число + unit). */
.order-materials-table__planfact {
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
  align-items: stretch;
  min-width: 0;
}

.order-materials-table__planfact-row {
  display: flex;
  justify-content: flex-end;
  gap: 0.4rem;
  align-items: baseline;
}

.order-materials-table__planfact-label {
  font-size: 0.68rem;
  color: var(--admin-muted);
  font-weight: 500;
  text-transform: uppercase;
  letter-spacing: 0.02em;
}

.order-materials-table__planfact-value {
  font-weight: 500;
}

.order-materials-table__planfact-delta {
  border-top: 1px dashed var(--admin-border, #e5e7eb);
  padding-top: 0.15rem;
  margin-top: 0.1rem;
  font-weight: 600;
}

.order-materials-table__planfact-delta--over .order-materials-table__planfact-value {
  color: var(--admin-danger-fg, #b91c1c);
}

.order-materials-table__planfact-delta--under .order-materials-table__planfact-value {
  color: var(--admin-success-fg, #047857);
}

.order-materials-table__planfact-delta--equal .order-materials-table__planfact-value {
  color: var(--admin-muted);
}

.order-materials-table__money {
  display: inline-flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 0.1rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

.order-materials-table__money--empty {
  color: var(--admin-muted);
}

.order-materials-table__warning {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  font-size: 0.7rem;
  color: var(--admin-warning-fg, #92400e);
  font-weight: 500;
  flex-wrap: wrap;
}

.order-materials-table__warning-cta {
  display: inline-flex;
  align-items: center;
  gap: 0.15rem;
  padding: 1px 6px;
  border-radius: 999px;
  background: rgba(245, 158, 11, 0.18);
  color: var(--admin-warning-fg, #92400e);
  font-size: 0.7rem;
  font-weight: 600;
  text-decoration: none;
  border: 1px solid rgba(245, 158, 11, 0.35);
}

.order-materials-table__warning-cta:hover {
  background: rgba(245, 158, 11, 0.28);
  text-decoration: none;
}

.order-materials-table__status {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  align-items: flex-start;
}

.order-materials-table__date {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

.order-materials-table__date--expected {
  color: var(--admin-muted);
}

.order-materials-table__date--received {
  color: var(--admin-text);
  font-weight: 500;
}

.order-materials-table__supplier {
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  min-width: 0;
}

.order-materials-table__supplier-name {
  font-weight: 500;
  word-break: break-word;
}

.order-materials-table__supplier-item {
  font-size: 0.72rem;
  word-break: break-word;
}

.order-materials-table__comment {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  max-width: 240px;
}

.order-materials-table__comment--empty {
  color: var(--admin-muted);
}

.order-materials-table__comment-text {
  font-size: 0.78rem;
  word-break: break-word;
  display: -webkit-box;
  -webkit-line-clamp: 2;
  -webkit-box-orient: vertical;
  overflow: hidden;
}

.order-materials-table__comment-warnings {
  margin: 0;
  padding: 0;
  list-style: none;
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
  font-size: 0.7rem;
  color: var(--admin-warning-fg, #92400e);
}

.order-materials-table__comment-warnings li {
  display: inline-flex;
  align-items: center;
  gap: 0.2rem;
}

/* Compact manual unblock block (под таблицей) */
.order-materials-manual-unlock {
  display: flex;
  flex-direction: column;
  gap: 0.45rem;
  background: var(--admin-surface);
  border: 1px dashed var(--admin-border);
  border-radius: 12px;
  padding: 0.6rem 0.9rem;
  min-width: 0;
}

.order-materials-manual-unlock__head {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  color: var(--admin-text);
}

.order-materials-manual-unlock__title {
  margin: 0;
  font-size: 0.88rem;
  font-weight: 600;
}

.order-materials-manual-unlock__hint {
  margin: 0;
  font-size: 0.78rem;
  color: var(--admin-muted);
  line-height: 1.4;
}

.order-materials-manual-unlock__empty {
  font-size: 0.8rem;
  padding: 0.2rem 0;
}

.order-materials-manual-unlock__overrides {
  border-top: 1px dashed var(--admin-border);
  padding-top: 0.4rem;
  font-size: 0.78rem;
}

.order-materials-manual-unlock__overrides > summary {
  cursor: pointer;
  list-style: revert;
  -webkit-user-select: none;
          user-select: none;
}

.order-materials-manual-unlock__overrides-list {
  margin: 0.4rem 0 0;
  padding: 0;
  list-style: none;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}

.order-materials-manual-unlock__overrides-item {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  padding: 0.4rem 0.5rem;
  background: var(--admin-warning-bg, #fff7ed);
  border: 1px solid var(--admin-warning-border, #fed7aa);
  border-radius: 6px;
}

.order-materials-manual-unlock__overrides-meta {
  font-size: 0.8rem;
}

.order-materials-manual-unlock__overrides-author {
  font-size: 0.72rem;
}

.order-materials-manual-unlock__overrides-comment {
  font-size: 0.78rem;
  font-style: italic;
}

@media (max-width: 720px) {
  .order-materials-table-card {
    padding: 0.55rem 0.65rem;
  }
  .order-materials-table-wrap .admin-table {
    min-width: 1100px;
  }
  .order-materials-table__comment {
    max-width: none;
  }
  /* На узком экране глобальный card-collapse делает ячейки блоками —
     sticky/фиксированная ширина 1-й и 2-й колонок тут только мешают. */
  .order-materials-table .admin-table th:first-child,
  .order-materials-table .admin-table td:first-child,
  .order-materials-table .admin-table th:nth-child(2),
  .order-materials-table .admin-table td:nth-child(2) {
    position: static;
    width: auto;
    min-width: 0;
    max-width: none;
    box-shadow: none;
  }
}

/* ---------------------------------------------------------------------------
   Unified order operations table — `/admin/orders/[id]` вкладка «Операции».
   Аналогично «Материалам»: одна compact-таблица + компактный итоговый
   блок снизу. Раньше во вкладке были три отдельные карточки
   («Маршрут», «План операций», «Производственная цепочка»), теперь —
   одна таблица со всеми операциями маршрута и итог под ней.
   Backend / Prisma / OperationPlan formulas / ProductionBalanceService
   не менялись.
   --------------------------------------------------------------------------- */

.order-operations-table-card {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  padding: 0.7rem 0.9rem 0.85rem;
  min-width: 0;
}

.order-operations-table-card__head {
  display: flex;
  align-items: center;
  gap: 0.6rem;
  flex-wrap: wrap;
}

.order-operations-table-card__title {
  font-size: 0.95rem;
  font-weight: 700;
  color: var(--admin-text);
  margin-right: 0.25rem;
}

.order-operations-table-card__summary {
  font-size: 0.78rem;
  color: var(--admin-muted);
  flex: 1 1 auto;
  min-width: 0;
}

.order-operations-table-card__errors {
  margin: 0;
  padding: 0.4rem 0.6rem;
  list-style: none;
  background: var(--admin-warning-bg, #fef3c7);
  border: 1px solid var(--admin-warning-border, #fcd34d);
  color: var(--admin-warning-fg, #92400e);
  border-radius: 6px;
  font-size: 0.78rem;
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
}

.order-operations-table-wrap {
  width: 100%;
  overflow-x: auto;
}

.order-operations-table {
  /* AdminTable монтирует <div class="admin-table-wrap"><table class="admin-table"/>.
     Здесь мы скоупим стили под наш режим: компактный padding,
     горизонтальный scroll на узких экранах, font 13px. */
  min-width: 1100px;
  font-size: 0.81rem;
}

.order-operations-table-wrap .admin-table {
  min-width: 1100px;
}

.order-operations-table-wrap .admin-table th,
.order-operations-table-wrap .admin-table td {
  padding: 0.4rem 0.55rem;
  vertical-align: top;
}

.order-operations-table__op-name {
  font-weight: 600;
  color: var(--admin-text);
  word-break: break-word;
}

.order-operations-table__status {
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  align-items: flex-start;
}

.order-operations-table__qty {
  display: inline-flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 0.1rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

.order-operations-table__money {
  display: inline-flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 0.1rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

.order-operations-table__money--empty {
  color: var(--admin-muted);
}

.order-operations-table__duration {
  display: inline-block;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

.order-operations-table__duration--empty {
  color: var(--admin-muted);
}

.order-operations-table__warning {
  display: inline-flex;
  align-items: center;
  gap: 0.2rem;
  font-size: 0.7rem;
  color: var(--admin-warning-fg, #92400e);
  font-weight: 500;
}

.order-operations-table__comment {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  max-width: 240px;
}

.order-operations-table__comment--empty {
  color: var(--admin-muted);
}

.order-operations-table__comment-text {
  font-size: 0.78rem;
  word-break: break-word;
  display: -webkit-box;
  -webkit-line-clamp: 2;
  -webkit-box-orient: vertical;
  overflow: hidden;
}

.order-operations-table__comment-warnings {
  margin: 0;
  padding: 0;
  list-style: none;
  display: flex;
  flex-direction: column;
  gap: 0.1rem;
}

/* Compact summary — итог под таблицей операций. */
.order-operations-summary {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
  padding: 0.55rem 0.6rem;
  border: 1px dashed var(--admin-border);
  border-radius: 10px;
  background: var(--admin-bg, #f8fafc);
  margin-top: 0.2rem;
}

.order-operations-summary__title {
  font-size: 0.85rem;
  font-weight: 600;
  color: var(--admin-text);
}

.order-operations-summary__items {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem 1.2rem;
}

.order-operations-summary__item {
  display: inline-flex;
  flex-direction: column;
  gap: 0.05rem;
  min-width: 0;
}

.order-operations-summary__label {
  font-size: 0.72rem;
  color: var(--admin-muted);
  text-transform: uppercase;
  letter-spacing: 0.02em;
}

.order-operations-summary__value {
  font-size: 0.88rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--admin-text);
}

.order-operations-summary__warning {
  display: flex;
  align-items: flex-start;
  gap: 0.35rem;
  padding: 0.4rem 0.55rem;
  border-radius: 6px;
  background: var(--admin-warning-bg, #fff7ed);
  color: var(--admin-warning-fg, #9a3412);
  font-size: 0.78rem;
  line-height: 1.35;
}

@media (max-width: 720px) {
  .order-operations-table-card {
    padding: 0.55rem 0.65rem;
  }
  .order-operations-table,
  .order-operations-table-wrap .admin-table {
    min-width: 980px;
  }
  .order-operations-table__comment {
    max-width: none;
  }
}

/* ---------------------------------------------------------------------------
   Unified order summary table — `/admin/orders/[id]` вкладка
   «Сводно по заказу». Одна общая таблица, отвечающая на вопрос
   «сколько стоит одна единица продукции с учётом всех расходов».
   Сверху — компактная KPI-полоса, снизу — итоговый блок
   (себестоимость / продажа / маржа). Backend / Prisma / OperationPlan
   / WorkshopNeed / OrderCostEstimate formulas НЕ менялись.
   --------------------------------------------------------------------------- */

.order-summary-table-card {
  display: flex;
  flex-direction: column;
  gap: 0.6rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 14px;
  padding: 0.7rem 0.9rem 0.85rem;
  min-width: 0;
}

.order-summary-table-card__errors {
  margin: 0;
  padding: 0.4rem 0.6rem;
  list-style: none;
  background: var(--admin-warning-bg, #fef3c7);
  border: 1px solid var(--admin-warning-border, #fcd34d);
  color: var(--admin-warning-fg, #92400e);
  border-radius: 6px;
  font-size: 0.78rem;
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
}

/* KPI bar — компактные plate-стиля карточки в одну строку. */
.order-summary-kpi-bar {
  display: flex;
  flex-wrap: wrap;
  gap: 0.4rem;
  padding: 0.45rem 0.55rem;
  background: var(--admin-bg, #f8fafc);
  border: 1px dashed var(--admin-border);
  border-radius: 10px;
}

.order-summary-kpi {
  display: inline-flex;
  flex-direction: column;
  gap: 0.05rem;
  min-width: 0;
  padding: 0.1rem 0.55rem;
  border-right: 1px solid var(--admin-border, #e2e8f0);
}

.order-summary-kpi:last-child {
  border-right: none;
}

.order-summary-kpi__label {
  font-size: 0.7rem;
  color: var(--admin-muted);
  text-transform: uppercase;
  letter-spacing: 0.02em;
  white-space: nowrap;
}

.order-summary-kpi__value {
  font-size: 0.92rem;
  font-weight: 600;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  color: var(--admin-text);
  white-space: nowrap;
}

/* Таблица. */
.order-summary-table-wrap {
  width: 100%;
  overflow-x: auto;
}

.order-summary-table {
  /* AdminTable монтирует <div class="admin-table-wrap"><table class="admin-table"/>.
     Скоупим стили под наш режим: компактный padding, font 13px,
     горизонтальный scroll. */
  min-width: 1100px;
  font-size: 0.81rem;
}

.order-summary-table-wrap .admin-table {
  min-width: 1100px;
}

.order-summary-table-wrap .admin-table th,
.order-summary-table-wrap .admin-table td {
  padding: 0.4rem 0.55rem;
  vertical-align: top;
}

.order-summary-table__section {
  display: inline-flex;
  align-items: center;
  font-size: 0.72rem;
  font-weight: 600;
  letter-spacing: 0.02em;
  text-transform: uppercase;
  padding: 0.1rem 0.45rem;
  border-radius: 999px;
  background: var(--admin-bg, #f1f5f9);
  color: var(--admin-text);
  white-space: nowrap;
}

.order-summary-table__section[data-section="OPERATION"] {
  background: #ecfeff;
  color: #155e75;
}

.order-summary-table__section[data-section="MATERIAL"] {
  background: #f0f9ff;
  color: #075985;
}

.order-summary-table__section[data-section="HARDWARE"] {
  background: #fef3c7;
  color: #92400e;
}

.order-summary-table__section[data-section="APPLICATION"] {
  background: #f3e8ff;
  color: #6b21a8;
}

.order-summary-table__section[data-section="OTHER"] {
  background: #f1f5f9;
  color: #475569;
}

.order-summary-table__article {
  font-weight: 500;
  color: var(--admin-text);
  word-break: break-word;
}

.order-summary-table__qty {
  display: inline-block;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
}

.order-summary-table__unit {
  display: inline-block;
  font-size: 0.78rem;
  color: var(--admin-muted);
  white-space: nowrap;
}

.order-summary-table__money {
  display: inline-flex;
  flex-direction: column;
  align-items: flex-end;
  gap: 0.1rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
  text-align: right;
}

.order-summary-table__money--empty {
  color: var(--admin-muted);
}

.order-summary-table__unit-cost {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
  font-weight: 500;
  text-align: right;
  display: inline-block;
}

.order-summary-table__unit-cost--empty {
  color: var(--admin-muted);
}

.order-summary-table__share {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  white-space: nowrap;
  display: inline-block;
  text-align: right;
}

.order-summary-table__share--empty {
  color: var(--admin-muted);
}

.order-summary-table__warning {
  display: inline-flex;
  align-items: center;
  gap: 0.2rem;
  font-size: 0.7rem;
  color: var(--admin-warning-fg, #92400e);
  font-weight: 500;
}

.order-summary-table__warning-list {
  margin: 0;
  padding: 0.4rem 0.6rem;
  list-style: none;
  display: flex;
  flex-direction: column;
  gap: 0.2rem;
  background: var(--admin-warning-bg, #fff7ed);
  color: var(--admin-warning-fg, #9a3412);
  border-radius: 6px;
  font-size: 0.78rem;
}

.order-summary-table__warning-list li {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
}

.order-summary-table__comment {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  max-width: 240px;
}

.order-summary-table__comment-text {
  font-size: 0.78rem;
  word-break: break-word;
  display: -webkit-box;
  -webkit-line-clamp: 2;
  -webkit-box-orient: vertical;
  overflow: hidden;
}

/* Итоговый блок под таблицей. */
.order-summary-totals {
  display: flex;
  flex-direction: column;
  gap: 0.55rem;
  padding: 0.55rem 0.65rem;
  border: 1px dashed var(--admin-border);
  border-radius: 10px;
  background: var(--admin-bg, #f8fafc);
}

.order-summary-totals__section {
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
}

.order-summary-totals__section-title {
  margin: 0;
  font-size: 0.78rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: var(--admin-muted);
}

.order-summary-totals__rows {
  margin: 0;
  padding: 0;
  display: grid;
  grid-template-columns: 1fr auto;
  row-gap: 0.2rem;
  column-gap: 0.8rem;
}

.order-summary-totals__row {
  display: contents;
}

.order-summary-totals__row dt,
.order-summary-totals__row .order-summary-totals__label {
  margin: 0;
  font-size: 0.83rem;
  color: var(--admin-text);
}

.order-summary-totals__row dd,
.order-summary-totals__row .order-summary-totals__value {
  margin: 0;
  font-size: 0.85rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  text-align: right;
  color: var(--admin-text);
  white-space: nowrap;
}

.order-summary-totals__row--total dt,
.order-summary-totals__row--total .order-summary-totals__label,
.order-summary-totals__row--total dd,
.order-summary-totals__row--total .order-summary-totals__value {
  font-weight: 700;
  font-size: 0.92rem;
}

.order-summary-margin--positive {
  color: var(--admin-success-fg, #047857);
}

.order-summary-margin--positive .order-summary-totals__value,
.order-summary-margin--positive .order-summary-kpi__value {
  color: var(--admin-success-fg, #047857);
}

.order-summary-margin--negative {
  color: var(--admin-danger-fg, #b91c1c);
}

.order-summary-margin--negative .order-summary-totals__value,
.order-summary-margin--negative .order-summary-kpi__value {
  color: var(--admin-danger-fg, #b91c1c);
}

@media (max-width: 720px) {
  .order-summary-table-card {
    padding: 0.55rem 0.65rem;
  }
  .order-summary-table,
  .order-summary-table-wrap .admin-table {
    min-width: 960px;
  }
  .order-summary-table__comment {
    max-width: none;
  }
  .order-summary-kpi {
    border-right: none;
    border-bottom: 1px dashed var(--admin-border, #e2e8f0);
    padding-bottom: 0.3rem;
  }
}

/* ============================================================================
   Order management view (`/admin/orders/[id]`, view-mode).
   --------------------------------------------------------------------------
   Стили для нового набора компонентов:
     - `OrderManagementHeader` (компактная управленческая шапка);
     - `OrderActionCenter` (короткие задачи / предупреждения);
     - вкладки `Производство / Паспорта / План / Потребности / История`.
   Линейка вкладок переиспользует существующий `.order-detail-tabs*`,
   общий wrapper `.order-workspace*` тоже не дублируется. Здесь только
   стили под локальные блоки.
   ============================================================================ */

/* --- Header -------------------------------------------------------------- */

.order-mgmt-header {
  /* Расширяем `.order-hero-card`-каркас, чтобы шапка вписывалась в
     общую сетку workspace-а и одинаково лежала на всех вкладках. */
  gap: 0.7rem;
}

.order-mgmt-header__grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
  gap: 0.7rem 1.1rem;
}

.order-mgmt-header__field {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  min-width: 0;
}

.order-mgmt-header__field-label {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  font-size: 0.74rem;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  color: var(--admin-muted);
}

.order-mgmt-header__field-icon {
  display: inline-flex;
  color: var(--admin-muted);
}

.order-mgmt-header__field-value {
  font-size: 0.95rem;
  color: var(--admin-text);
  display: inline-flex;
  flex-wrap: wrap;
  align-items: baseline;
  gap: 0.35rem;
  min-width: 0;
}

.order-mgmt-header__field-hint {
  font-size: 0.78rem;
  color: var(--admin-muted);
}

.order-mgmt-header__unit {
  font-size: 0.78rem;
  color: var(--admin-muted);
  font-weight: 500;
}

.order-mgmt-header__deadline {
  display: inline-flex;
  flex-wrap: wrap;
  gap: 0.4rem;
  align-items: center;
}

.order-mgmt-header__progress {
  display: inline-flex;
  flex-direction: column;
  gap: 0.3rem;
  min-width: 100px;
  width: 100%;
}

.order-mgmt-header__progress .admin-deadline-progress {
  height: 6px;
  width: 100%;
  background: rgba(15, 23, 42, 0.08);
  border-radius: 4px;
  overflow: hidden;
}

.order-mgmt-header__progress .admin-deadline-progress__bar {
  height: 100%;
  background: var(--admin-accent, #2563eb);
  border-radius: 4px;
}

.order-mgmt-header__warning {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.45rem 0.6rem;
  border-radius: 8px;
  background: var(--admin-warning-bg, #fef3c7);
  color: var(--admin-warning-fg, #92400e);
  font-size: 0.82rem;
  border: 1px solid var(--admin-warning-border, #fcd34d);
  align-self: flex-start;
}

.order-mgmt-header__actions {
  flex-wrap: wrap;
  gap: 0.5rem;
}

.order-mgmt-header__actions-hint {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  font-size: 0.85rem;
  color: var(--admin-muted);
  padding: 0.3rem 0.5rem;
}

/* --- Action center ------------------------------------------------------ */

.order-action-center {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  padding: 0.85rem 1rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 16px;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04);
}

.order-action-center__head {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 0.6rem;
  flex-wrap: wrap;
}

.order-action-center__title {
  margin: 0;
  font-size: 0.95rem;
  font-weight: 700;
  color: var(--admin-text);
}

.order-action-center__count {
  font-size: 0.78rem;
  color: var(--admin-muted);
  background: rgba(15, 23, 42, 0.04);
  padding: 0.1rem 0.5rem;
  border-radius: 999px;
}

.order-action-center__empty {
  margin: 0;
  font-size: 0.85rem;
  color: var(--admin-muted);
}

.order-action-center__list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
}

.order-action-center__item {
  display: flex;
  align-items: flex-start;
  gap: 0.55rem;
  padding: 0.55rem 0.65rem;
  border-radius: 10px;
  background: rgba(15, 23, 42, 0.03);
  border-left: 3px solid var(--admin-border);
}

.order-action-center__item--info {
  background: var(--admin-info-bg, #dbeafe);
  border-left-color: var(--admin-info-border, #93c5fd);
}

.order-action-center__item--warning {
  background: var(--admin-warning-bg, #fef3c7);
  border-left-color: var(--admin-warning-border, #fcd34d);
}

.order-action-center__item--danger {
  background: var(--admin-danger-bg, #fee2e2);
  border-left-color: var(--admin-danger-border, #fca5a5);
}

.order-action-center__item-icon {
  flex: 0 0 auto;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 22px;
  height: 22px;
  color: var(--admin-muted);
}

.order-action-center__item--info .order-action-center__item-icon {
  color: var(--admin-info-fg, #1e3a8a);
}

.order-action-center__item--warning .order-action-center__item-icon {
  color: var(--admin-warning-fg, #9a3412);
}

.order-action-center__item--danger .order-action-center__item-icon {
  color: var(--admin-danger-fg, #b91c1c);
}

.order-action-center__item-body {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  min-width: 0;
  flex: 1 1 auto;
}

.order-action-center__item-title {
  font-size: 0.9rem;
  font-weight: 600;
  color: var(--admin-text);
}

.order-action-center__item-hint {
  font-size: 0.8rem;
  color: var(--admin-muted);
}

.order-action-center__item-cta {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  font-size: 0.82rem;
  color: var(--admin-accent, #2563eb);
  text-decoration: none;
  border-radius: 6px;
  padding: 0.2rem 0.4rem;
  white-space: nowrap;
}

.order-action-center__item-cta:hover {
  text-decoration: underline;
}

/* --- Production tab ----------------------------------------------------- */

.order-prod-tab {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
}

.order-prod-tab__kpi-grid {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
  gap: 0.5rem;
}

.order-prod-tab__kpi {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  padding: 0.55rem 0.7rem;
  border-radius: 10px;
  background: rgba(15, 23, 42, 0.03);
  border: 1px solid var(--admin-border);
}

.order-prod-tab__kpi--success {
  background: var(--admin-success-bg, #dcfce7);
  border-color: var(--admin-success-border, #bbf7d0);
}

.order-prod-tab__kpi--warning {
  background: var(--admin-warning-bg, #fef3c7);
  border-color: var(--admin-warning-border, #fcd34d);
}

.order-prod-tab__kpi--danger {
  background: var(--admin-danger-bg, #fee2e2);
  border-color: var(--admin-danger-border, #fca5a5);
}

.order-prod-tab__kpi-label {
  font-size: 0.74rem;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  font-weight: 600;
  color: var(--admin-muted);
}

.order-prod-tab__kpi-value {
  font-size: 1.2rem;
  font-weight: 700;
  color: var(--admin-text);
}

.order-prod-tab__shopfloor-warning {
  margin: 0 0 0.6rem;
  font-size: 0.85rem;
  color: var(--admin-muted);
}

/* --- Объединённая матрица «размер × этап» -------------------------------- */

.order-prod-matrix__scroll {
  overflow-x: auto;
}

.order-prod-matrix {
  width: 100%;
  border-collapse: collapse;
  font-size: 0.84rem;
  white-space: nowrap;
}

.order-prod-matrix th,
.order-prod-matrix td {
  padding: 0.35rem 0.55rem;
  border-bottom: 1px solid var(--admin-border);
  text-align: right;
}

.order-prod-matrix__num {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}

.order-prod-matrix__size {
  position: sticky;
  left: 0;
  z-index: 1;
  text-align: left;
  font-weight: 700;
  background: var(--admin-card-bg, #fff);
}

/* Шапка групп */
.order-prod-matrix__group-row th {
  text-align: center;
  font-size: 0.72rem;
  text-transform: uppercase;
  letter-spacing: 0.04em;
  color: var(--admin-muted);
  border-bottom: 1px solid var(--admin-border);
}

.order-prod-matrix__group {
  background: rgba(15, 23, 42, 0.03);
}

.order-prod-matrix__group--wip {
  background: var(--admin-danger-bg, #fee2e2);
}

.order-prod-matrix__group--ship {
  background: var(--admin-success-bg, #dcfce7);
}

.order-prod-matrix__legend {
  display: inline-block;
  margin-left: 0.4rem;
  font-size: 0.66rem;
  font-weight: 500;
  letter-spacing: 0;
  text-transform: none;
  opacity: 0.7;
}

.order-prod-matrix__sub-row th {
  font-size: 0.72rem;
  font-weight: 600;
  color: var(--admin-muted);
  border-bottom: 2px solid var(--admin-border);
}

/* Граница между группами колонок */
.order-prod-matrix__group-end {
  border-right: 2px solid var(--admin-border);
}

/* Хвост воронки — отгрузка */
.order-prod-matrix__finished {
  font-weight: 600;
  color: #15803d;
}

.order-prod-matrix__ready {
  color: #1d4ed8;
}

/* Ячейки «В цеху сейчас» */
.order-prod-matrix__wip {
  text-align: center;
}

.order-prod-matrix__wip--foot {
  background: rgba(15, 23, 42, 0.02);
}

.order-prod-matrix__wip-empty {
  color: var(--admin-muted);
  opacity: 0.5;
}

.order-prod-matrix__wip-chips {
  display: inline-flex;
  flex-direction: column;
  gap: 0.15rem;
  align-items: stretch;
}

.order-prod-matrix__chip {
  display: inline-block;
  padding: 0.05rem 0.35rem;
  border-radius: 6px;
  font-size: 0.76rem;
  font-weight: 700;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  line-height: 1.3;
}

.order-prod-matrix__chip--now {
  background: var(--admin-danger-bg, #fee2e2);
  color: #b91c1c;
}

.order-prod-matrix__chip--done {
  background: var(--admin-success-bg, #dcfce7);
  color: #15803d;
}

.order-prod-matrix__totals th,
.order-prod-matrix__totals td {
  border-top: 2px solid var(--admin-border);
  border-bottom: none;
  font-weight: 700;
  background: rgba(15, 23, 42, 0.03);
}

/* --- Plan tab ----------------------------------------------------------- */

.order-plan-tab {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
}

.order-plan-tab__grid {
  display: grid;
  grid-template-columns: minmax(0, 2fr) minmax(0, 1fr);
  gap: var(--admin-space-md);
  align-items: start;
}

.order-plan-tab__col-main,
.order-plan-tab__col-aside {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
  min-width: 0;
}

@media (max-width: 1199px) {
  .order-plan-tab__grid {
    grid-template-columns: minmax(0, 1fr);
  }
}

/* --- Route mode toggle (адаптивный сплит-распошив) --------------------- */

.order-plan-tab__route-mode {
  display: flex;
  flex-direction: column;
  gap: 4px;
  margin: var(--admin-space-sm) 0;
}

.admin-deflist__subtitle {
  font-size: 0.8rem;
  color: var(--admin-text-muted, #6b7280);
}

.route-mode-toggle__row {
  display: inline-flex;
  gap: 4px;
  padding: 3px;
  border-radius: 8px;
  background: var(--admin-surface-muted, #f1f5f9);
  width: -moz-fit-content;
  width: fit-content;
}

.route-mode-toggle__btn {
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 0;
  padding: 4px 12px;
  border: 1px solid transparent;
  border-radius: 6px;
  background: transparent;
  color: var(--admin-text-muted, #6b7280);
  cursor: pointer;
  line-height: 1.15;
}

.route-mode-toggle__btn:hover:not(:disabled) {
  background: var(--admin-surface, #fff);
}

.route-mode-toggle__btn--active {
  background: var(--admin-surface, #fff);
  border-color: var(--admin-border, #d1d5db);
  color: var(--admin-text, #111827);
  cursor: default;
}

.route-mode-toggle__btn-label {
  font-size: 0.85rem;
  font-weight: 600;
}

.route-mode-toggle__btn-hint {
  font-size: 0.68rem;
  opacity: 0.8;
}

/* --- Passports tab ------------------------------------------------------ */

.order-passports-tab__filters {
  display: grid;
  grid-template-columns: minmax(220px, 2fr) repeat(2, minmax(140px, 1fr)) auto;
  gap: 0.6rem 1rem;
  align-items: end;
  margin-bottom: 0.6rem;
}

.order-passports-tab__search {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  background: var(--admin-surface);
  border: 1px solid var(--admin-border);
  border-radius: 8px;
  padding: 0.1rem 0.4rem;
}

.order-passports-tab__search input {
  border: none;
  outline: none;
  background: transparent;
  width: 100%;
  font-size: 0.85rem;
  padding: 0.2rem 0;
}

@media (max-width: 720px) {
  .order-passports-tab__filters {
    grid-template-columns: minmax(0, 1fr);
  }
}

/* --- Outsource list ----------------------------------------------------- */

.order-outsource-list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}

.order-outsource-list__item {
  padding: 0.55rem 0.7rem;
  border-radius: 10px;
  background: rgba(15, 23, 42, 0.02);
  border: 1px solid var(--admin-border);
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
}

.order-outsource-list__head {
  display: flex;
  align-items: baseline;
  gap: 0.45rem;
  flex-wrap: wrap;
}

.order-outsource-list__name {
  font-size: 0.92rem;
  color: var(--admin-text);
}

/* --- History tab -------------------------------------------------------- */

.order-history-tab {
  display: flex;
  flex-direction: column;
  gap: var(--admin-space-md);
}

.order-history-tab__passport-list {
  list-style: none;
  margin: 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
}

.order-history-tab__passport-item {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 0.5rem;
  padding: 0.4rem 0.5rem;
  border-radius: 8px;
  background: rgba(15, 23, 42, 0.02);
  border: 1px solid var(--admin-border);
}

/* =====================================================================
   Admin UI 2.6 — карточка паспорта `/admin/passports/[id]`.

   Стили только для новой admin-страницы паспорта (см.
   `apps/web/app/admin/passports/[id]/page.tsx`). На legacy
   `/passports/[id]` ничего не задеваем — там по-прежнему живут
   `passport-hero`, `passport-fields` и `qc-summary` из старой
   эпохи.
   ===================================================================== */

.admin-link {
  color: var(--admin-primary, #1d4ed8);
  text-decoration: underline;
  text-underline-offset: 2px;
}
.admin-link:hover {
  color: var(--admin-primary-strong, #1e40af);
}

.admin-passport-fields {
  display: grid;
  grid-template-columns: 1fr;
  gap: 0;
}

.admin-passport-field {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
  gap: 1rem;
  padding: 0.55rem 0;
  border-bottom: 1px solid var(--admin-border, #e5e7eb);
}
.admin-passport-field:last-child {
  border-bottom: 0;
}
.admin-passport-field__label {
  color: var(--admin-muted, #64748b);
  font-size: 0.88rem;
  flex: 0 0 auto;
}
.admin-passport-field__value {
  text-align: right;
  font-size: 0.95rem;
}

.admin-passport-qr {
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 0.65rem;
}
.admin-passport-qr img {
  width: 180px;
  height: 180px;
  border-radius: 12px;
  border: 1px solid var(--admin-border, #e5e7eb);
  background: #fff;
  padding: 8px;
}
.admin-passport-qr__code {
  font-size: 0.78rem;
  color: var(--admin-muted, #64748b);
  letter-spacing: 0.08em;
}

.admin-passport-qc {
  display: grid;
  grid-template-columns: repeat(3, minmax(0, 1fr));
  gap: 0.6rem;
}
.admin-passport-qc__cell {
  display: flex;
  flex-direction: column;
  gap: 0.15rem;
  padding: 0.6rem 0.7rem;
  border-radius: 10px;
  background: rgba(15, 23, 42, 0.03);
  border: 1px solid var(--admin-border, #e5e7eb);
}
.admin-passport-qc__cell strong {
  font-size: 1.15rem;
}
.admin-passport-qc__cell--bad {
  background: rgba(220, 38, 38, 0.06);
  border-color: rgba(220, 38, 38, 0.2);
}
.admin-passport-qc__cell--bad strong {
  color: #b91c1c;
}
.admin-passport-qc__cell--good {
  background: rgba(22, 163, 74, 0.06);
  border-color: rgba(22, 163, 74, 0.2);
}
.admin-passport-qc__cell--good strong {
  color: #15803d;
}

.admin-passport-earnings {
  overflow-x: auto;
}
.admin-passport-earnings table {
  width: 100%;
  border-collapse: collapse;
}
.admin-passport-earnings th {
  text-align: left;
  font-weight: 600;
  font-size: 0.82rem;
  color: var(--admin-muted, #64748b);
  text-transform: uppercase;
  letter-spacing: 0.04em;
  padding: 0.45rem 0.5rem;
  border-bottom: 1px solid var(--admin-border, #e5e7eb);
}
.admin-passport-earnings td {
  padding: 0.55rem 0.5rem;
  border-bottom: 1px solid var(--admin-border, #e5e7eb);
  font-size: 0.92rem;
}
.admin-passport-earnings tbody tr:last-child td {
  border-bottom: 0;
}

/* ==========================================================================
   Доски кабинетов раскройщика и конструктора (`apps/web/app/{cutter,
   constructor}/`). Рабочий экран собирается общим шаблоном
   `TerminalShell` (обёртка `.work .work--seamstress`, как у ОТК), ниже —
   только классы самих досок задач.
   ========================================================================== */
.constructor-page__title {
  font-size: 1.25rem;
  margin: 0 0 1rem;
  font-weight: 600;
}
.constructor-section {
  margin-bottom: 1.5rem;
}
.constructor-section__title {
  font-size: 0.95rem;
  font-weight: 600;
  margin: 0 0 0.5rem;
  display: flex;
  align-items: baseline;
  gap: 0.4rem;
}
.constructor-section__count {
  color: #64748b;
  font-weight: 400;
}
.constructor-section__empty {
  color: #64748b;
  font-size: 0.9rem;
  margin: 0;
  padding: 0.75rem;
  border: 1px dashed #cbd5e1;
  border-radius: 6px;
  background: #fff;
}

.constructor-list {
  list-style: none;
  padding: 0;
  margin: 0;
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.constructor-card {
  display: block;
  padding: 0.75rem 0.9rem;
  background: #fff;
  border: 1px solid #e2e8f0;
  border-radius: 8px;
  text-decoration: none;
  color: inherit;
}
.constructor-card:hover {
  border-color: #94a3b8;
}
.constructor-card__head {
  display: flex;
  justify-content: space-between;
  gap: 0.5rem;
  align-items: baseline;
  margin-bottom: 0.4rem;
}
.constructor-card__name {
  font-size: 1rem;
}
.constructor-card__article {
  color: #64748b;
  font-size: 0.8rem;
}
.constructor-card__meta {
  display: grid;
  grid-template-columns: repeat(2, 1fr);
  gap: 0.25rem 0.75rem;
  margin: 0;
  font-size: 0.85rem;
}
.constructor-card__meta dt {
  color: #64748b;
  display: inline;
  margin-right: 4px;
}
.constructor-card__meta dd {
  display: inline;
  margin: 0;
}
.constructor-card__meta > div {
  display: flex;
  gap: 4px;
}
.constructor-card__comment {
  margin: 0.5rem 0 0;
  padding-top: 0.5rem;
  border-top: 1px solid #e2e8f0;
  font-size: 0.85rem;
  color: #334155;
  white-space: pre-wrap;
}

/* Цветовой акцент карточки по статусу — чтобы статус был «выделен
   цветом» во всех секциях одинаково заметно (а не только бейджем).
   Реализован через inset box-shadow слева: насыщенный цвет (тон `-fg`),
   без изменения геометрии (вёрстка не сдвигается). */
.constructor-card {
  box-shadow: inset 4px 0 0 0 transparent;
}
.constructor-card--status-new,
.constructor-card--status-pending_accept {
  box-shadow: inset 4px 0 0 0 var(--admin-warning-fg);
}
.constructor-card--status-in_progress {
  box-shadow: inset 4px 0 0 0 var(--admin-primary-fg);
}
.constructor-card--status-rework {
  box-shadow: inset 4px 0 0 0 var(--admin-danger-fg);
}
.constructor-card--status-done {
  box-shadow: inset 4px 0 0 0 var(--admin-success-fg);
}
.constructor-card--status-cancelled {
  box-shadow: inset 4px 0 0 0 var(--admin-muted);
}

/* Drag-and-drop «взять в работу» в кабинете конструктора.
   Перетаскиваются только карточки из «Свободные»/«Доработка»
   (.constructor-card--draggable); приёмник — секция «В работе у меня»
   (.constructor-section--dropzone). Подсветка через outline, чтобы не
   двигать вёрстку при drag-over. */
.constructor-card--draggable {
  cursor: grab;
}
.constructor-card--draggable:active {
  cursor: grabbing;
}
.constructor-section--drop-active {
  outline: 2px dashed var(--admin-primary, #2563eb);
  outline-offset: 4px;
  border-radius: 8px;
  background: var(--admin-primary-soft);
}
.constructor-section--drop-busy {
  opacity: 0.6;
  pointer-events: none;
}

.constructor-detail {
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.constructor-back {
  font-size: 0.85rem;
  color: #2563eb;
  text-decoration: none;
}
.constructor-back:hover {
  text-decoration: underline;
}
.constructor-detail__head {
  display: flex;
  justify-content: space-between;
  gap: 1rem;
  align-items: flex-start;
}
.constructor-detail__title {
  margin: 0;
  font-size: 1.25rem;
  font-weight: 600;
}
.constructor-detail__article {
  color: #64748b;
  font-size: 0.85rem;
  margin-top: 2px;
}

/* Бейдж статуса в кабинете конструктора. Цвета берутся из тех же
   `--admin-*-soft / --admin-*-fg` токенов, что и `AdminStatusBadge`,
   чтобы статус выглядел одинаково и в админке, и в кабинете
   конструктора (источник истины тонов — `CONSTRUCTOR_TASK_STATUS_TONE`
   в `@sewing/shared/constructor-tasks`). Геометрия (padding/font/uppercase)
   подобрана под mobile-first кабинет — крупнее, чем admin-pill. */
.constructor-status {
  font-size: 0.78rem;
  font-weight: 600;
  padding: 0.2rem 0.65rem;
  border-radius: var(--admin-radius-pill);
  letter-spacing: 0.02em;
  white-space: nowrap;
  /* Рамка под цвет статуса (currentColor = `--admin-*-fg` из варианта).
     Без неё бледные фоны NEW/IN_PROGRESS (warning-soft/primary-soft) на
     белой карточке почти сливаются и статус кажется «без цвета». */
  border: 1px solid currentColor;
  display: inline-block;
}
.constructor-status--new,
.constructor-status--pending_accept {
  background: var(--admin-warning-soft);
  color: var(--admin-warning-fg);
}
.constructor-status--in_progress {
  background: var(--admin-primary-soft);
  color: var(--admin-primary-fg);
}
.constructor-status--rework {
  background: var(--admin-danger-soft);
  color: var(--admin-danger-fg);
}
.constructor-status--done {
  background: var(--admin-success-soft);
  color: var(--admin-success-fg);
}
.constructor-status--cancelled {
  background: var(--admin-soft);
  color: var(--admin-muted);
}

.constructor-card-block {
  background: #fff;
  border: 1px solid #e2e8f0;
  border-radius: 8px;
  padding: 0.85rem 1rem;
}
.constructor-card-block--complete {
  border-color: #fecaca;
  background: #fff7f7;
}
.constructor-card-block__title {
  margin: 0 0 0.5rem;
  font-size: 0.95rem;
  font-weight: 600;
}
.constructor-meta-grid {
  display: grid;
  grid-template-columns: repeat(2, 1fr);
  gap: 0.4rem 1rem;
  margin: 0;
  font-size: 0.9rem;
}
.constructor-meta-grid dt {
  color: #64748b;
  font-size: 0.78rem;
  margin: 0;
}
.constructor-meta-grid dd {
  margin: 0;
}

.constructor-muted {
  color: #64748b;
  font-size: 0.9rem;
  margin: 0;
}
.constructor-link {
  color: #2563eb;
  text-decoration: none;
}
.constructor-link:hover {
  text-decoration: underline;
}
.constructor-comment-view {
  margin: 0;
  white-space: pre-wrap;
  font-size: 0.92rem;
  color: #1f2937;
}

.constructor-table {
  width: 100%;
  border-collapse: collapse;
  font-size: 0.9rem;
}
.constructor-table th,
.constructor-table td {
  padding: 0.45rem 0.6rem;
  text-align: left;
  border-bottom: 1px solid #e2e8f0;
}
.constructor-table th {
  color: #64748b;
  font-weight: 500;
  font-size: 0.78rem;
  text-transform: uppercase;
  letter-spacing: 0.04em;
}
.constructor-table tbody tr:last-child td {
  border-bottom: 0;
}
/* Таблица «завершить задачу» с колонкой DXF-файла. На десктопе —
   обычная таблица; на узких экранах native `input[type=file]` не
   сжимается и распирает строку, поэтому на ≤600px разворачиваем каждую
   строку в карточку-стек (label сверху, поле на всю ширину). */
.constructor-table-wrap {
  max-width: 100%;
}
.constructor-table input[type='file'] {
  max-width: 100%;
}
@media (max-width: 600px) {
  /* Стек только для редактируемой формы (обёрнута в
     `.constructor-table-wrap`), read-only «Размерную таблицу» не
     трогаем — она узкая и так помещается. */
  .constructor-table-wrap .constructor-table thead {
    display: none;
  }
  .constructor-table-wrap .constructor-table,
  .constructor-table-wrap .constructor-table tbody,
  .constructor-table-wrap .constructor-table tr,
  .constructor-table-wrap .constructor-table td {
    display: block;
  }
  .constructor-table-wrap .constructor-table tr {
    border: 1px solid #e2e8f0;
    border-radius: 8px;
    padding: 0.5rem 0.7rem;
    margin-bottom: 0.6rem;
  }
  .constructor-table-wrap .constructor-table td {
    border-bottom: 0;
    padding: 0.3rem 0;
  }
  /* Подпись колонки над полем (берётся из data-label ячейки). */
  .constructor-table-wrap .constructor-table td::before {
    content: attr(data-label);
    display: block;
    margin-bottom: 0.2rem;
    font-size: 0.72rem;
    font-weight: 600;
    text-transform: uppercase;
    letter-spacing: 0.04em;
    color: #64748b;
  }
  /* «Размер» — заголовок карточки, без подписи. */
  .constructor-table-wrap .constructor-table td[data-label='Размер'] {
    font-size: 1.05rem;
    margin-bottom: 0.25rem;
  }
  .constructor-table-wrap .constructor-table td[data-label='Размер']::before {
    display: none;
  }
  /* Поля на всю ширину карточки — file-input больше не вылезает. */
  .constructor-table-wrap .constructor-table input[type='text'],
  .constructor-table-wrap .constructor-table input[type='file'] {
    width: 100% !important;
    max-width: 100%;
  }
}

/* Кабинет раскройщика (`apps/web/app/cutter/*`) — переиспользует
   `.constructor-*` chrome; здесь только специфика интерактивных таблиц
   раскроя (редактируемые поля, живые итоги, управление рулонами). */
.cutter-form {
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
.cutter-table tfoot td {
  border-top: 2px solid #cbd5e1;
  font-weight: 600;
  color: #1f2937;
}
.cutter-input {
  width: 5.5rem;
  max-width: 100%;
  padding: 0.35rem 0.5rem;
  border: 1px solid #cbd5e1;
  border-radius: 6px;
  font-size: 0.95rem;
}
.cutter-input:focus {
  outline: none;
  border-color: #2563eb;
  box-shadow: 0 0 0 2px rgba(37, 99, 235, 0.15);
}
.cutter-total {
  text-align: right;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  font-weight: 600;
}
.cutter-total--reached {
  color: #16a34a;
}
.cutter-roll-remove {
  padding: 0.2rem 0.55rem;
  line-height: 1;
}
.cutter-actions {
  flex-direction: row;
  flex-wrap: wrap;
  gap: 0.6rem;
}
.cutter-layers-total {
  margin: 0.6rem 0;
  font-size: 1rem;
  color: #1f2937;
}
.cutter-layers-total strong {
  font-size: 1.15rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}

.constructor-files {
  list-style: none;
  padding: 0;
  margin: 0;
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
  font-size: 0.9rem;
}

.constructor-actions {
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
  align-items: flex-start;
}
.constructor-actions__error {
  color: #b91c1c;
  font-size: 0.85rem;
  margin: 0;
}
.constructor-actions__saved {
  color: #166534;
  font-size: 0.85rem;
}

.constructor-btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: 0.4rem;
  padding: 0.55rem 1rem;
  border-radius: 6px;
  border: 1px solid transparent;
  cursor: pointer;
  font-weight: 600;
  font-size: 0.9rem;
}
.constructor-btn:disabled {
  opacity: 0.6;
  cursor: not-allowed;
}
.constructor-btn--primary {
  background: #2563eb;
  color: #fff;
}
.constructor-btn--primary:hover:not(:disabled) {
  background: #1d4ed8;
}
.constructor-btn--ghost {
  background: #fff;
  color: #1f2937;
  border-color: #cbd5e1;
}
.constructor-btn--ghost:hover:not(:disabled) {
  background: #f8fafc;
}

.constructor-comment-form {
  margin-top: 0.75rem;
  display: flex;
  flex-direction: column;
  gap: 0.4rem;
}
.constructor-comment-form__footer {
  display: flex;
  align-items: center;
  gap: 0.75rem;
}
.constructor-label {
  font-weight: 500;
  font-size: 0.85rem;
  color: #334155;
}
.constructor-textarea {
  width: 100%;
  padding: 0.55rem 0.6rem;
  border: 1px solid #cbd5e1;
  border-radius: 6px;
  font-family: inherit;
  font-size: 0.9rem;
  resize: vertical;
}
.constructor-textarea:focus {
  outline: 2px solid #93c5fd;
  border-color: #2563eb;
}

.constructor-complete-form {
  display: flex;
  flex-direction: column;
  gap: 0.75rem;
}
.constructor-complete-form__hint {
  margin: 0;
  font-size: 0.85rem;
  color: #475569;
}
.constructor-complete-form__rows {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
}
.constructor-complete-form__row {
  display: grid;
  grid-template-columns: 1fr 2fr;
  gap: 0.5rem;
  align-items: center;
  padding: 0.45rem 0.6rem;
  background: #fff;
  border: 1px solid #e2e8f0;
  border-radius: 6px;
}
.constructor-complete-form__row input[type='file'] {
  font-size: 0.85rem;
}

@media (max-width: 600px) {
  .constructor-card__meta {
    grid-template-columns: 1fr;
  }
  .constructor-meta-grid {
    grid-template-columns: 1fr;
  }
  .constructor-complete-form__row {
    grid-template-columns: 1fr;
  }
  .constructor-detail__head {
    flex-direction: column;
    align-items: flex-start;
  }
}

/* ===========================================================================
 * Кабинет мастера: вкладки + «Доска движения тиража»
 * (apps/web/app/master/production-board-view.tsx,
 *  packages/shared/src/production-board.ts)
 * ======================================================================== */

.master-page__tabs {
  display: flex;
  gap: 6px;
  margin: 4px 0 12px;
}
.master-page__tab {
  padding: 9px 16px;
  border-radius: var(--radius-pill);
  border: 1px solid transparent;
  background: transparent;
  color: var(--color-fg-muted);
  font-size: 14px;
  font-weight: 600;
  cursor: pointer;
}
.master-page__tab:hover {
  background: var(--color-bg-muted);
}
.master-page__tab.is-active {
  background: var(--color-accent);
  color: #fff;
}

.pboard {
  display: flex;
  flex-direction: column;
  gap: 12px;
}
.pboard__muted {
  color: var(--color-fg-muted);
}
.pboard__bar {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 8px;
  font-size: 13px;
}
.pboard__period,
.pboard__refresh,
.pboard__expand {
  border: 1px solid var(--color-border-strong);
  background: var(--color-bg-card);
  color: var(--color-fg-muted);
  border-radius: var(--radius-md);
  padding: 5px 12px;
  font-size: 13px;
  font-weight: 600;
  cursor: pointer;
}
/* Тогл «Растянуть/Свернуть» имеет смысл только на десктопе:
   мобайл уже рендерит аккордеон вместо таблицы (см. медиазапрос ниже). */
.pboard__expand {
  display: none;
}
.pboard__expand[aria-pressed='true'] {
  background: var(--color-accent);
  border-color: var(--color-accent);
  color: #fff;
}
.pboard__period.is-active {
  background: var(--color-accent);
  border-color: var(--color-accent);
  color: #fff;
}
.pboard__refresh {
  margin-left: auto;
}
.pboard__refresh:disabled {
  opacity: 0.6;
  cursor: default;
}

.pboard__date {
  font-weight: 700;
  font-size: 13px;
}
.pboard__order {
  font-size: 10.5px;
  color: var(--color-fg-subtle);
  margin-top: 2px;
  line-height: 1.3;
}
.pboard__recon {
  margin-top: 6px;
  border-top: 1px solid var(--color-border);
  padding-top: 6px;
  display: flex;
  flex-direction: column;
  gap: 4px;
}
.pboard__recon-line {
  font-size: 11px;
  line-height: 1.35;
}
.pboard__chip {
  align-self: flex-start;
  display: inline-flex;
  align-items: center;
  gap: 4px;
  padding: 2px 7px;
  border-radius: var(--radius-pill);
  font-size: 10.5px;
  font-weight: 700;
}
.pboard__chip--bad {
  background: var(--color-danger-soft);
  color: var(--color-danger-fg);
}
.pboard__chip--ok {
  background: var(--color-ok-soft);
  color: var(--color-ok-fg);
}

.pboard__cell {
  width: 100%;
  text-align: left;
  cursor: pointer;
  background: var(--color-bg-muted);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  padding: 8px 10px;
  min-height: 62px;
  color: var(--color-fg);
}
.pboard__cell:hover {
  border-color: var(--color-accent);
  background: var(--color-bg-tint);
}
.pboard__cell--empty {
  background: transparent;
  border: 1px dashed var(--color-border);
  cursor: default;
  color: var(--color-fg-subtle);
  display: flex;
  align-items: center;
  justify-content: center;
}
.pboard__cell--released {
  background: var(--color-ok-soft);
  border-color: var(--color-ok-soft);
  color: var(--color-ok-fg);
}
.pboard__cell-top {
  font-size: 14px;
  font-weight: 700;
  display: flex;
  align-items: center;
  gap: 6px;
}
/* «Дошло N» — главная цифра ячейки в накопительной модели партии
   (см. `ProductionBoardStageBucketDto.received` shared-DTO). */
.pboard__cell-received {
  font-weight: 700;
}
/* «Выпущено N · сейчас K» — вторая строка ячейки. «Сейчас» приходит
   через `bucket.passports`, см. `production-board-view.tsx`.
   Цветовая семантика по просьбе мастера (21.05.2026):
     - «выпущено» — зелёным, как KPI «уже сдано с шага»;
     - «сейчас»   — красным, как сигнал «висит, нужно следить». */
.pboard__cell-flow {
  font-size: 12px;
  color: var(--color-fg);
  margin-top: 2px;
  line-height: 1.4;
}
.pboard__cell-released-num {
  color: var(--color-ok-fg);
  font-weight: 700;
}
.pboard__cell-here-num {
  color: var(--color-danger-fg);
  font-weight: 700;
}
.pboard__cell-emps {
  font-size: 12px;
  color: var(--color-fg-muted);
  margin-top: 3px;
  line-height: 1.4;
  display: flex;
  flex-wrap: wrap;
  gap: 4px;
}
/* Плашка-сотрудник в превью ячейки. Active — нейтральная,
   released — зелёная (✔ закрыл операцию). Раздельные плашки
   для одного и того же человека возможны: «Акмар·25» (active)
   + «✔ Акмар·10» (released) — он одновременно и шьёт, и часть
   уже сдал. */
.pboard__emp-chip {
  display: inline-flex;
  align-items: center;
  padding: 1px 6px;
  border-radius: var(--radius-sm);
  background: var(--color-bg-tint);
  color: var(--color-fg-muted);
  white-space: nowrap;
}
.pboard__emp-chip--released {
  background: var(--color-ok-soft);
  color: var(--color-ok-fg);
  font-weight: 700;
}
.pboard__cell-more {
  font-size: 12px;
  font-weight: 600;
  color: var(--color-accent);
  margin-top: 3px;
}
.pboard__released-qty {
  font-size: 14px;
  font-weight: 700;
}
.pboard__released-pct {
  font-size: 11px;
  opacity: 0.8;
  margin-top: 2px;
}
.pboard__released-strong {
  font-size: 12px;
  font-weight: 700;
  color: var(--color-ok-fg);
}
.pboard__badge-def {
  display: inline-block;
  margin-left: 6px;
  padding: 0 6px;
  border-radius: 6px;
  background: var(--color-danger-soft);
  color: var(--color-danger-fg);
  font-size: 11px;
  font-weight: 700;
}
.pboard__def-text {
  color: var(--color-danger-fg);
}

.pboard__desktop {
  overflow-x: auto;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  background: var(--color-bg-card);
}
.pboard__table {
  border-collapse: collapse;
  width: 100%;
  min-width: 1120px;
}
.pboard__table th {
  text-align: left;
  font-size: 12.5px;
  font-weight: 700;
  color: var(--color-fg-muted);
  padding: 12px 14px;
  background: var(--color-bg-muted);
  white-space: nowrap;
}
.pboard__table td {
  padding: 10px 14px;
  vertical-align: top;
  border-top: 1px solid var(--color-border);
}
.pboard__rowhead {
  position: sticky;
  left: 0;
  z-index: 2;
  background: var(--color-bg-card);
  min-width: 158px;
  max-width: 174px;
}
.pboard__table thead .pboard__rowhead {
  background: var(--color-bg-muted);
}
.pboard__th-released {
  background: var(--color-ok-soft) !important;
  color: var(--color-ok-fg) !important;
}

/* Режим «растянуть» (см. `production-board-view.tsx`, кнопка `.pboard__expand`).
   Только десктоп — мобайл всё равно прячет `.pboard__desktop` (см. ниже).
   `.master-page` шириной 720px центрирован через `margin: 0 auto`, поэтому
   приём `left: 50%; transform: translateX(-50%)` относительно родителя
   совпадает с центром вьюпорта; ширина капается 1600px, чтобы на
   ультравайдах матрица не растягивалась до неудобства. */
@media (min-width: 861px) {
  .pboard__expand {
    display: inline-flex;
    align-items: center;
  }
  .pboard--wide .pboard__desktop {
    position: relative;
    left: 50%;
    transform: translateX(-50%);
    width: min(calc(100vw - 2rem), 1600px);
  }
  .pboard--wide .pboard__table {
    min-width: 100%;
  }
}

.pboard__mobile {
  display: none;
  flex-direction: column;
  gap: 10px;
}
.pboard__acc {
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  background: var(--color-bg-card);
  overflow: hidden;
}
.pboard__acc-head {
  width: 100%;
  display: flex;
  justify-content: space-between;
  align-items: flex-start;
  gap: 10px;
  padding: 13px 16px;
  background: transparent;
  border: 0;
  cursor: pointer;
  text-align: left;
}
.pboard__caret {
  color: var(--color-fg-subtle);
}
.pboard__acc-body {
  border-top: 1px solid var(--color-border);
  padding: 12px;
  display: flex;
  flex-direction: column;
  gap: 8px;
}
.pboard__stagecard {
  background: var(--color-bg-muted);
  border-radius: var(--radius-md);
  padding: 12px;
}
.pboard__stagecard--released {
  background: var(--color-ok-soft);
}
.pboard__stagecard-h {
  display: flex;
  justify-content: space-between;
  align-items: center;
  margin-bottom: 8px;
  font-size: 14px;
  font-weight: 700;
}

@media (max-width: 860px) {
  .pboard__desktop {
    display: none;
  }
  .pboard__mobile {
    display: flex;
  }
}

.pboard__overlay {
  position: fixed;
  inset: 0;
  z-index: 50;
  background: rgba(2, 6, 23, 0.42);
  display: flex;
  align-items: flex-end;
  justify-content: center;
}
.pboard__panel {
  width: 480px;
  max-width: 100%;
  max-height: 86vh;
  background: var(--color-bg-card);
  border-radius: var(--radius-lg) var(--radius-lg) 0 0;
  display: flex;
  flex-direction: column;
  box-shadow: 0 -10px 40px rgba(2, 6, 23, 0.3);
}
@media (min-width: 861px) {
  .pboard__overlay {
    align-items: stretch;
    justify-content: flex-end;
  }
  .pboard__panel {
    height: 100%;
    max-height: 100%;
    border-radius: 0;
  }
}
.pboard__panel-head {
  display: flex;
  justify-content: space-between;
  align-items: flex-start;
  padding: 14px 16px;
  border-bottom: 1px solid var(--color-border);
}
.pboard__panel-title {
  font-weight: 700;
}
.pboard__panel-sub {
  font-size: 12px;
  color: var(--color-fg-muted);
  margin-top: 2px;
}
.pboard__x {
  border: 0;
  background: transparent;
  color: var(--color-fg-muted);
  font-size: 16px;
  cursor: pointer;
  padding: 4px 8px;
  border-radius: 8px;
}
.pboard__x:hover {
  background: var(--color-bg-muted);
}
.pboard__panel-body {
  overflow-y: auto;
  padding: 14px;
}
.pboard__emp-sec {
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  margin-bottom: 10px;
  overflow: hidden;
}
/* Released-группа в дрилл-панели: ✔ <Имя> = «он уже закрыл операцию,
   паспорта ждут следующего исполнителя». Зелёная шапка целиком — отличить
   от активной работы (нейтральный фон). См.
   `production-board-view.tsx::ProductionBoardDrillEmployeeGroupDto.released`. */
.pboard__emp-sec--released {
  border-color: var(--color-ok-soft);
}
.pboard__emp-sec--released .pboard__emp-sec-h {
  background: var(--color-ok-soft);
  color: var(--color-ok-fg);
}
.pboard__emp-sec-h {
  display: flex;
  justify-content: space-between;
  align-items: center;
  padding: 9px 12px;
  background: var(--color-bg-muted);
  font-weight: 600;
  font-size: 14px;
}
.pboard__pp {
  display: flex;
  justify-content: space-between;
  align-items: center;
  padding: 8px 12px;
  border-top: 1px solid var(--color-border);
  font-size: 13px;
}
.pboard__qty {
  display: inline-block;
  min-width: 46px;
  text-align: center;
  padding: 1px 6px;
  border-radius: 6px;
  background: var(--color-accent-soft);
  color: var(--color-accent-fg);
  font-weight: 700;
  font-size: 12px;
  margin-right: 8px;
}

/* -----------------------------------------------------------------------
 * Кабинет мастера → вкладка «Сотрудники» (статистика «кто сколько
 * сделал»). См. `apps/web/app/master/employee-stats-view.tsx`,
 * `packages/shared/src/master-employee-stats.ts`. Overlay/панель drill
 * переиспользуют `.pboard__overlay`/`.pboard__panel`.
 * ----------------------------------------------------------------------- */

.mstat__bar {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 8px;
  margin: 10px 0 12px;
}
.mstat__presets {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: 6px;
}
.mstat__dates {
  display: flex;
  align-items: center;
  gap: 8px;
  margin-left: auto;
}
.mstat__date-field {
  display: inline-flex;
  align-items: center;
  gap: 4px;
  font-size: 13px;
}
.mstat__date-field input[type='date'] {
  padding: 5px 8px;
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  font-size: 13px;
  background: var(--color-bg);
  color: inherit;
}

.mstat__table {
  width: 100%;
  border-collapse: collapse;
  font-size: 13px;
}
.mstat__table th {
  text-align: left;
  padding: 6px 8px;
  border-bottom: 2px solid var(--color-border);
  color: var(--color-text-muted, #6b7280);
  font-weight: 600;
  white-space: nowrap;
}
.mstat__table td {
  padding: 8px;
  border-bottom: 1px solid var(--color-border);
  vertical-align: middle;
}
.mstat__table th.mstat__num,
.mstat__table td.mstat__num {
  text-align: right;
  white-space: nowrap;
  width: 1%;
}
.mstat__row {
  cursor: pointer;
}
.mstat__row:hover,
.mstat__row:focus-visible {
  background: var(--color-bg-muted);
  outline: none;
}
.mstat__name {
  font-weight: 600;
}
.mstat__role {
  font-size: 11px;
  color: var(--color-text-muted, #6b7280);
}
.mstat__td-ops {
  max-width: 360px;
}
.mstat__ops {
  display: flex;
  flex-wrap: wrap;
  gap: 4px;
}
.mstat__op-chip {
  display: inline-flex;
  align-items: center;
  padding: 1px 7px;
  border-radius: 6px;
  background: var(--color-accent-soft);
  color: var(--color-accent-fg);
  font-size: 12px;
  white-space: nowrap;
}
.mstat__op-def {
  color: var(--color-danger-fg);
  font-weight: 700;
  margin-left: 3px;
}
.mstat__op-more {
  font-size: 11px;
  color: var(--color-text-muted, #6b7280);
  align-self: center;
}
.mstat__def {
  color: var(--color-danger-fg);
  font-weight: 700;
}

.mstat__drill-section {
  margin-bottom: 14px;
}
.mstat__drill-h {
  font-size: 12px;
  font-weight: 700;
  text-transform: uppercase;
  letter-spacing: 0.03em;
  color: var(--color-text-muted, #6b7280);
  margin-bottom: 6px;
}
.mstat__drill-op,
.mstat__drill-day {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 10px;
  padding: 7px 2px;
  border-top: 1px solid var(--color-border);
  font-size: 13px;
}
.mstat__drill-op-name {
  font-weight: 600;
}

/* -----------------------------------------------------------------------
 * /admin/employees: row actions + danger zone + hard-delete modal
 * (см. `apps/web/app/admin/employees/row-actions.tsx`,
 * `apps/web/app/admin/employees/[id]/danger-zone.tsx`,
 * `docs/employee-deletion-recon.md`).
 * ----------------------------------------------------------------------- */

.employee-row-actions {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  white-space: nowrap;
}

.employee-row-actions__wrap {
  position: relative;
  display: inline-flex;
}

.employee-row-actions__inline {
  display: inline-flex;
  align-items: center;
  gap: 0.35rem;
  padding: 0.3rem 0.6rem;
  border: 1px solid var(--admin-border);
  border-radius: 6px;
  background: #fff;
  color: inherit;
  font-size: 0.85rem;
  cursor: pointer;
  transition: background 0.15s, border-color 0.15s, color 0.15s;
}
.employee-row-actions__inline:hover:not(:disabled),
.employee-row-actions__inline:focus-visible {
  background: #f1f5f9;
  border-color: #cbd5e1;
  outline: none;
}
.employee-row-actions__inline:disabled {
  opacity: 0.5;
  cursor: not-allowed;
}

.employee-row-actions__error {
  position: absolute;
  top: calc(100% + 4px);
  right: 0;
  min-width: 220px;
  padding: 0.4rem 0.55rem;
  font-size: 0.82rem;
  background: var(--admin-danger-soft, #fee2e2);
  color: var(--admin-danger-fg, #991b1b);
  border: 1px solid #fecaca;
  border-radius: 6px;
  z-index: 21;
}

/* Danger zone -------------------------------------------------------------- */

.employee-danger-zone {
  display: flex;
  flex-direction: column;
  gap: 0.9rem;
  margin-top: 1rem;
  padding: 1rem 1.1rem;
  border: 1px solid #fecaca;
  border-radius: var(--radius-xl, 12px);
  background: #fffafa;
}
.employee-danger-zone__head {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  color: var(--admin-danger-fg, #991b1b);
}
.employee-danger-zone__title {
  margin: 0;
  font-size: 1rem;
  font-weight: 700;
}
.employee-danger-zone__row {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 1rem;
  padding: 0.7rem 0;
  border-top: 1px dashed #fecaca;
}
.employee-danger-zone__row:first-of-type {
  border-top: 0;
  padding-top: 0;
}
.employee-danger-zone__text {
  flex: 1 1 auto;
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
  font-size: 0.92rem;
}
.employee-danger-zone__text strong {
  font-size: 0.98rem;
}
.employee-danger-zone__text p {
  margin: 0;
}
.employee-danger-zone__blockers {
  margin: 0.2rem 0 0;
  padding-left: 1.1rem;
  color: var(--admin-danger-fg, #991b1b);
  font-size: 0.85rem;
}
.employee-danger-zone__hint {
  margin: 0.3rem 0 0;
  font-size: 0.82rem;
  color: var(--admin-muted);
}
.employee-danger-zone__error {
  padding: 0.5rem 0.7rem;
  background: var(--admin-danger-soft, #fee2e2);
  color: var(--admin-danger-fg, #991b1b);
  border: 1px solid #fecaca;
  border-radius: 6px;
  font-size: 0.88rem;
}

/* Hard-delete modal -------------------------------------------------------- */

.hard-delete-modal__card {
  max-width: 460px;
  gap: 0.9rem;
}
.hard-delete-modal__summary {
  display: grid;
  grid-template-columns: max-content 1fr;
  gap: 0.3rem 0.9rem;
  margin: 0;
  padding: 0.6rem 0.8rem;
  background: #f8fafc;
  border: 1px solid var(--admin-border);
  border-radius: 8px;
  font-size: 0.9rem;
}
.hard-delete-modal__summary dt {
  font-weight: 600;
  color: var(--admin-muted);
}
.hard-delete-modal__summary dd {
  margin: 0;
}
.hard-delete-modal__ack {
  display: flex;
  align-items: flex-start;
  gap: 0.5rem;
  font-size: 0.88rem;
  color: var(--admin-danger-fg, #991b1b);
}
.hard-delete-modal__ack input {
  margin-top: 0.2rem;
}
.hard-delete-modal__field {
  display: flex;
  flex-direction: column;
  gap: 0.35rem;
}
.hard-delete-modal__field label {
  font-size: 0.88rem;
  font-weight: 500;
}
.hard-delete-modal__field input {
  padding: 0.55rem 0.7rem;
  font-size: 0.95rem;
  border: 1px solid var(--admin-border);
  border-radius: 6px;
}
.hard-delete-modal__field input:focus {
  outline: 2px solid var(--admin-danger, #dc2626);
  outline-offset: -1px;
  border-color: var(--admin-danger, #dc2626);
}
.hard-delete-modal__actions {
  display: flex;
  justify-content: flex-end;
  gap: 0.5rem;
}

/* -----------------------------------------------------------------------
 * /packing: сворачиваемые секции «Открытые коробки» и «Ждут размещения»
 * на Stage 1 терминала упаковщика. Шапка кликабельна целиком, кнопка-
 * trigger без рамки — выглядит как заголовок с каретой.
 * См. `apps/web/app/packing/packing-terminal.tsx`.
 * ----------------------------------------------------------------------- */
.packing-section-toggle {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  background: transparent;
  border: 0;
  padding: 0;
  margin: 0;
  cursor: pointer;
  color: inherit;
  text-align: left;
}
.packing-section-toggle:focus-visible {
  outline: 2px solid var(--color-accent, #4f46e5);
  outline-offset: 2px;
  border-radius: 4px;
}
.packing-section-toggle__caret {
  display: inline-flex;
  width: 1rem;
  justify-content: center;
  color: var(--color-fg-muted, #64748b);
  font-size: 0.85rem;
  line-height: 1;
}

/* -----------------------------------------------------------------------
 * «Мой день» — чип в `.employee-toolbar` + панель + история.
 *
 * Цель — единая точка входа сотрудника в свою дневную статистику:
 * выработка (штуки), сумма начислений и пометка PENDING. Чип сидит
 * рядом с «Мастер» и «Мой QR-код» в правом столбике; в отличие от
 * круглых icon-only кнопок, у чипа сохраняется ширина под цифры,
 * поэтому override `.employee-toolbar .my-day-chip` НЕ режет его
 * до 40×40 (см. `.employee-toolbar .call-master-btn` выше).
 * ----------------------------------------------------------------------- */
.my-day-chip {
  pointer-events: auto;
  display: inline-flex;
  align-items: center;
  gap: 0.45rem;
  padding: 0.5rem 0.85rem;
  border: 0;
  border-radius: 999px;
  background: #1e293b;
  color: #ffffff;
  font-size: 0.85rem;
  font-weight: 700;
  letter-spacing: 0.01em;
  cursor: pointer;
  box-shadow: 0 4px 12px rgba(15, 23, 42, 0.28);
  transition: background 0.15s ease, transform 0.1s ease;
  white-space: nowrap;
}
.my-day-chip:hover:not(:disabled) {
  background: #0f172a;
}
.my-day-chip:active:not(:disabled) {
  transform: scale(0.97);
}
.my-day-chip__icon {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 1.3rem;
  height: 1.3rem;
  border-radius: 50%;
  background: rgba(255, 255, 255, 0.18);
  font-weight: 800;
  font-size: 0.9rem;
}
.my-day-chip__label { white-space: nowrap; }

/* Карточка модалки «Мой день». */
.my-day-panel__card {
  max-width: 380px;
  gap: 0.9rem;
}
.my-day-panel__section {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  padding: 0.75rem 0;
  border-top: 1px solid #e5e7eb;
}
.my-day-panel__section:first-of-type { border-top: 0; padding-top: 0; }
.my-day-panel__section-title {
  font-size: 0.8rem;
  font-weight: 700;
  letter-spacing: 0.04em;
  text-transform: uppercase;
  color: #64748b;
}
.my-day-panel__empty {
  color: #64748b;
  font-size: 0.9rem;
}

.my-day-panel__op-list {
  list-style: none;
  padding: 0;
  margin: 0;
  display: flex;
  flex-direction: column;
  gap: 0.3rem;
}
.my-day-panel__op-row {
  display: grid;
  grid-template-columns: 1fr auto auto;
  gap: 0.6rem;
  align-items: baseline;
  font-size: 0.95rem;
}
.my-day-panel__op-name { color: #0f172a; font-weight: 500; }
.my-day-panel__op-qty {
  color: #64748b;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  font-size: 0.85rem;
}
.my-day-panel__op-amount {
  color: #0f172a;
  font-weight: 600;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.my-day-panel__op-summary {
  display: flex;
  justify-content: space-between;
  align-items: baseline;
  padding-top: 0.5rem;
  border-top: 1px dashed #e5e7eb;
  font-size: 0.9rem;
  color: #475569;
}
.my-day-panel__op-summary-total {
  font-weight: 700;
  color: #0f172a;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.my-day-panel__pending {
  display: flex;
  gap: 0.4rem;
  align-items: flex-start;
  padding: 0.55rem 0.7rem;
  border-radius: 8px;
  background: #fff7ed;
  color: #9a3412;
  font-size: 0.85rem;
}

.my-day-panel__shift {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  font-size: 0.9rem;
  color: #334155;
}
.my-day-panel__shift-dot {
  width: 0.55rem;
  height: 0.55rem;
  border-radius: 50%;
  background: #16a34a;
  box-shadow: 0 0 0 3px rgba(22, 163, 74, 0.18);
}
.my-day-panel__shift-empty { color: #94a3b8; }
.my-day-panel__salary-amount {
  font-size: 0.95rem;
  color: #0f172a;
}
.my-day-panel__salary-amount strong {
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
  margin-left: 0.25rem;
}
.my-day-panel__salary-hint {
  color: #94a3b8;
  font-size: 0.85rem;
  margin-left: 0.4rem;
}

.my-day-panel__total {
  display: flex;
  justify-content: space-between;
  align-items: baseline;
  padding: 0.65rem 0.8rem;
  margin-top: 0.25rem;
  border-radius: 10px;
  background: #0f172a;
  color: #ffffff;
}
.my-day-panel__total-label {
  font-weight: 600;
  font-size: 0.9rem;
}
.my-day-panel__total-value {
  font-weight: 800;
  font-size: 1.1rem;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.my-day-panel__history-btn { margin-top: 0.25rem; }

/* История — модалка с таблицей. */
.my-history__card {
  max-width: 560px;
  gap: 0.6rem;
}
.my-history__table {
  width: 100%;
  border-collapse: collapse;
  font-size: 0.88rem;
}
.my-history__table th,
.my-history__table td {
  padding: 0.45rem 0.5rem;
  border-bottom: 1px solid #f1f5f9;
  text-align: left;
}
.my-history__table thead th {
  font-weight: 700;
  color: #64748b;
  font-size: 0.78rem;
  letter-spacing: 0.03em;
  text-transform: uppercase;
  border-bottom: 1px solid #e5e7eb;
}
.my-history__th-num,
.my-history__td-num {
  text-align: right;
  font-feature-settings: "tnum";
  font-variant-numeric: tabular-nums;
}
.my-history__th-status,
.my-history__td-status {
  text-align: center;
  width: 2.5rem;
}
.my-history__row--empty td { color: #cbd5e1; }
.my-history__td-total { font-weight: 700; color: #0f172a; }
.my-history__table tfoot td {
  border-top: 2px solid #0f172a;
  border-bottom: 0;
  font-weight: 700;
  color: #0f172a;
}

.my-history__chip {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  min-width: 1.6rem;
  height: 1.4rem;
  padding: 0 0.4rem;
  border-radius: 999px;
  font-size: 0.75rem;
  font-weight: 700;
}
.my-history__chip--ok { background: #dcfce7; color: #166534; }
.my-history__chip--pending { background: #ffedd5; color: #9a3412; }
.my-history__chip--off { background: #f1f5f9; color: #94a3b8; font-weight: 500; }

/* На узких экранах — сжимаем колонки таблицы, чтобы влезало. */
@media (max-width: 480px) {
  .my-history__card { max-width: 100%; }
  .my-history__table { font-size: 0.82rem; }
  .my-history__table th,
  .my-history__table td { padding: 0.35rem 0.3rem; }
}

/* ===========================================================================
 * Роли доступа сотрудника (фича «несколько ролей», /admin/employees/[id]).
 * Список чекбоксов всех assignable-ролей + radio «основная».
 * ======================================================================== */
.admin-roles__list {
  display: grid;
  grid-template-columns: repeat(auto-fill, minmax(220px, 1fr));
  gap: 0.35rem 1rem;
  margin-top: 0.4rem;
}
.admin-roles__row {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 0.75rem;
  padding: 0.25rem 0.1rem;
  border-bottom: 1px solid var(--admin-border, rgba(148, 163, 184, 0.18));
}
.admin-roles__check,
.admin-roles__primary {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  cursor: pointer;
  font-size: 0.9rem;
}
.admin-roles__primary {
  font-size: 0.8rem;
  white-space: nowrap;
}
.admin-roles__check input:disabled + span,
.admin-roles__primary input:disabled + span {
  opacity: 0.5;
  cursor: not-allowed;
}

/* ===========================================================================
 * «Сменить рабочее место» — плавающая кнопка для мульти-ролевых
 * сотрудников (фича «несколько ролей»). Сидит слева снизу, над
 * мобильной навигацией; на десктопе просто в углу.
 * ======================================================================== */
.workplace-switch-fab {
  position: fixed;
  left: 12px;
  bottom: 76px;
  z-index: 60;
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.5rem 0.85rem;
  border: none;
  border-radius: 999px;
  background: #0f172a;
  color: #fff;
  font-size: 0.85rem;
  font-weight: 600;
  box-shadow: 0 6px 18px rgba(15, 23, 42, 0.28);
  cursor: pointer;
}
.workplace-switch-fab:disabled { opacity: 0.6; cursor: default; }

.workplace-switch-confirm__backdrop {
  position: fixed;
  inset: 0;
  z-index: 120;
  display: flex;
  align-items: center;
  justify-content: center;
  background: rgba(15, 23, 42, 0.5);
  padding: 1rem;
}
.workplace-switch-confirm {
  background: var(--admin-card-bg, #fff);
  color: var(--admin-fg, #0f172a);
  border-radius: 14px;
  padding: 1.25rem;
  max-width: 360px;
  width: 100%;
  box-shadow: 0 20px 48px rgba(15, 23, 42, 0.32);
}
.workplace-switch-confirm__text { margin: 0 0 1rem; font-size: 0.95rem; }
.workplace-switch-confirm__actions {
  display: flex;
  justify-content: flex-end;
  gap: 0.5rem;
  flex-wrap: wrap;
}
.workplace-switch-toast {
  position: fixed;
  left: 50%;
  bottom: 88px;
  transform: translateX(-50%);
  z-index: 130;
  max-width: 90vw;
  padding: 0.6rem 1rem;
  border-radius: 10px;
  background: #b91c1c;
  color: #fff;
  font-size: 0.88rem;
  box-shadow: 0 8px 22px rgba(185, 28, 28, 0.35);
  cursor: pointer;
}

