/* cocheras.css — chrome propio de cocheras (NO Capa A/B compartida).

   Orden de <link> canónico (services/_shared-ui, "brief de estética" —
   ver docs/recon-briefs.md), el mismo que reports/meetings:

       fonts.css → tokens.css → base.css → motion.css → components.css
       → cocheras.css (overrides locales, ESTE archivo)
       → polish.css (gana la cascada sobre cocheras.css)
       → colors.css → palette.css → css/components/toolbar.css → mobile.css

   print.css NO se copió (fuera de la lista de la ola A) — se suma cuando
   haga falta un layout de impresión (planilla/reportes, ola D+).

   Nota para quien retome esto: MES (services/mes/app/static/css/mes.css)
   documenta el criterio INVERSO — su chrome local carga DESPUÉS de
   polish.css a propósito, para que gane la cascada sobre los defaults de
   polish (radios, botones). Acá seguimos el orden que cita diseno.md
   ("Orden de <link> canónico del brief de estética" = el de reports), pero
   si el QA visual de la ola A/C/D encuentra que polish.css pisa algo
   estructural del app-shell/sidebar, mover este <link> a después de
   mobile.css (como mes.css) es el fix de una línea.

   Contenido: SOLO el chrome de las dos superficies (app-shell mobile +
   sidebar desktop). Nada de componentes de negocio (eso vive en templates
   + CSS de página cuando llegue la ola C/D). */

/* ── Mobile: app-shell (services/mes/app/static/css/mes.css como referencia)
   .app = 100vh fijo (NUNCA 100dvh/100svh, cortos en cold-start iOS; NUNCA
   medido por JS, WebKit bug 170595). El nav es el ÚLTIMO hijo flex EN
   FLUJO NORMAL — NUNCA position:fixed (flota a media pantalla en iOS
   standalone). El glass va DIRECTO sobre el elemento (nunca en un
   ::before{z-index:-1}, iOS muestrea un backdrop vacío ahí). ────────────── */
.app {
    height: 100vh;
    display: flex;
    flex-direction: column;
    position: relative;
    background: var(--bg);
    color: var(--fg);
}

.app__header {
    flex: 0 0 auto;
    display: flex;
    align-items: center;
    justify-content: space-between;
    gap: var(--space-2);
    padding: max(var(--space-3), var(--safe-area-top)) var(--space-3) var(--space-2);
    border-bottom: 1px solid var(--border);
}

.app__brand {
    font-weight: 700;
    font-size: var(--text-lg);
    letter-spacing: -0.01em;
}

/* Genérico: cualquier <span class="accent"> en el chrome (brand mobile,
   brand sidebar, acceso.html) toma el color de marca. */
.accent {
    color: var(--accent);
}

.app__main {
    flex: 1 1 auto;
    overflow-y: auto;
    -webkit-overflow-scrolling: touch;
    padding: var(--space-3);
    padding-bottom: var(--space-4);
}

/* Bottom-nav: EN FLUJO (flex, no fixed). Glass bajo + brightness (patrón
   mes --card-glass-blur): blur fuerte se ve peor sobre OLED que un blur
   sutil + brillo. Par -webkit siempre. */
.nav {
    flex: 0 0 auto;
    display: flex;
    justify-content: space-around;
    gap: var(--space-1);
    padding: var(--space-2) var(--space-2) max(var(--space-2), var(--safe-area-bottom));
    background: var(--glass-bg);
    -webkit-backdrop-filter: blur(3px) brightness(1.1);
    backdrop-filter: blur(3px) brightness(1.1);
    border-top: 1px solid var(--border);
    isolation: isolate;
}

.nav__btn {
    flex: 1 1 0;
    display: flex;
    flex-direction: column;
    align-items: center;
    gap: 2px;
    min-height: 44px;
    padding: var(--space-1);
    background: transparent;
    border: none;
    border-radius: var(--radius-btn);
    color: var(--fg-muted);
    font-size: var(--text-xs);
    cursor: pointer;
    transition: color var(--dur-base) var(--ease);
}

.nav__btn--act {
    color: var(--accent);
}

/* ── Desktop: sidebar fija estilo reports (fuente: _shared-ui/sidebar.html).
   Ola A dejó el esqueleto; ola D la completa: sticky full-height + pie con
   chip de TC/link mobile/salir, ver más abajo "Ola D — desktop". ────────── */
.layout {
    display: flex;
    min-height: 100vh;
    background: var(--bg);
    color: var(--fg);
}

.sidebar {
    flex: 0 0 240px;
    width: 240px;
    display: flex;
    flex-direction: column;
    gap: var(--space-4);
    padding: var(--space-4) var(--space-3);
    background: var(--bg-card);
    border-right: 1px solid var(--border);
    /* Sticky full-height: el sidebar NO debe crecer con el contenido largo
       de /d/planilla o /d/clientes/{id} (antes heredaba la altura de .main
       vía flex-stretch, quedando gigante); overflow-y propio para no perder
       ítems de nav si algún día la lista crece más que 100vh. */
    position: sticky;
    top: 0;
    height: 100vh;
    overflow-y: auto;
}

.sidebar .brand {
    display: block;
    font-weight: 700;
    font-size: var(--text-lg);
    letter-spacing: -0.01em;
    color: var(--fg);
    text-decoration: none;
}

.sidebar nav ul {
    list-style: none;
    margin: 0;
    padding: 0;
    display: flex;
    flex-direction: column;
    gap: var(--space-1);
}

.sidebar nav a {
    display: flex;
    align-items: center;
    gap: var(--space-2);
    padding: var(--space-2) var(--space-3);
    border-radius: var(--radius-btn);
    color: var(--fg-muted);
    text-decoration: none;
    font-size: var(--text-sm);
    transition: background var(--dur-base) var(--ease), color var(--dur-base) var(--ease);
}

.sidebar nav a .icon {
    width: 18px;
    height: 18px;
    flex: 0 0 auto;
}

.sidebar nav a:hover {
    background: var(--bg-soft);
    color: var(--fg);
}

.sidebar nav a.active {
    background: var(--accent-bg-soft);
    color: var(--accent);
}

/* Pie del sidebar: chip de TC + link mobile + salir. margin-top:auto lo
   ancla abajo cuando el nav es corto (siempre, con 8 ítems); si algún día
   creciera, el overflow-y del .sidebar lo scrollea junto con el nav. */
.sidebar__foot {
    margin-top: auto;
    display: flex;
    flex-direction: column;
    gap: var(--space-2);
    padding-top: var(--space-3);
    border-top: 1px solid var(--border);
}

.tc-chip {
    padding: var(--space-2) var(--space-3);
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
    font-size: var(--text-xs);
}
.tc-chip__row {
    display: flex;
    align-items: baseline;
    justify-content: space-between;
    gap: var(--space-2);
}
.tc-chip__label {
    color: var(--fg-muted);
}
.tc-chip__valor {
    font-weight: 700;
}
.tc-chip__fecha {
    display: block;
    margin-top: 2px;
    color: var(--fg-faint);
}
.tc-chip__aviso {
    color: var(--err);
    font-weight: 600;
}
/* "Desactualizado" = viejo (no es un error de negocio, sólo una fuente
   externa que no respondió) — ámbar de aviso, no rojo de error. */
.tc-chip--viejo {
    border-color: color-mix(in srgb, #eab308 45%, var(--border));
}
.tc-chip--viejo .tc-chip__fecha {
    color: #eab308;
}

.sidebar__link {
    font-size: var(--text-sm);
    color: var(--fg-muted);
    text-decoration: none;
    padding: var(--space-1) var(--space-1);
}
.sidebar__link:hover {
    color: var(--fg);
}
.sidebar__salir {
    margin: 0;
}
.sidebar__salir .btn {
    width: 100%;
    justify-content: center;
}

.main {
    flex: 1 1 auto;
    min-width: 0;
    padding: var(--space-4) var(--space-5);
}

.main .content {
    max-width: 1200px;
}


/* ════════════════════════════════════════════════════════════════════
   Ola C — mobile: superficie completa (/m/*). Todo lo de acá abajo es
   EXCLUSIVO de mobile.css… no, de ESTE archivo (cocheras.css sigue siendo
   el único con el chrome de negocio, ver cabecera). Glass SUTIL (patrón
   MES): blur bajo + brightness, nunca blur fuerte solo.
   ════════════════════════════════════════════════════════════════════ */

/* ── Banner de conexión (dentro de .app, EN FLUJO — nunca fixed) ────────
   Reusa .banner/.banner__icon/.banner__msg de components.css; el único
   agregado acá es que quede compacto y sin margin (ya está dentro del
   flujo flex de .app, no necesita respirar como un banner de página).
   Selector `.banner.app__conn-banner` (2 clases) a propósito: polish.css
   redefine `.banner{border-radius,padding}` con la MISMA especificidad de
   un selector `.app__conn-banner` solo (0,0,1,0) y carga DESPUÉS en el
   `<link>` de base_m.html — ganaría la cascada y el banner volvería a
   verse como card redondeada en vez de tira full-width. 2 clases = 0,0,2,0,
   gana sin pelear con el orden de carga. */
.banner.app__conn-banner {
    flex: 0 0 auto;
    margin: 0;
    border-radius: 0;
    border-left: none;
    border-right: none;
    justify-content: center;
    text-align: center;
}
/* GOTCHA real (detectado en QA visual, no cosmético): `.banner{display:
   flex}` de components.css/polish.css es una regla de AUTHOR stylesheet —
   le gana SIEMPRE al `[hidden]{display:none}` del UA stylesheet, sin
   importar la especificidad (el origen pesa antes que la especificidad en
   la cascada). Resultado sin este fix: el banner de conexión se veía
   SIEMPRE (el ícono "!" asomando bajo el header) aunque tuviera el
   atributo `hidden` puesto — bug real, no hipotético. ID selector fuerza
   el `display:none` por encima de cualquier `.banner{display:flex}`. */
#banner-conexion[hidden] {
    display: none;
}

/* ── page-head: título + volver. Vive AL PRINCIPIO de `main` en cada
   página con navegación hacia atrás (cliente, forms). Las raíces de nav
   (Inicio, Más) usan `.page-head--root` (sin back-chevron). ───────────── */
.page-head {
    display: flex;
    align-items: center;
    gap: var(--space-2);
    margin-bottom: var(--space-3);
    min-height: 44px;
}
.page-head--root {
    min-height: 0;
}
.page-head__back {
    display: inline-flex;
    align-items: center;
    justify-content: center;
    width: 36px;
    height: 36px;
    flex: 0 0 auto;
    margin-left: calc(-1 * var(--space-2));
    border-radius: var(--radius-btn);
    color: var(--fg-muted);
    transition: color var(--dur-base) var(--ease), background var(--dur-base) var(--ease);
}
.page-head__back:hover,
.page-head__back:focus-visible {
    color: var(--fg);
    background: var(--bg-soft);
}
.page-head__title {
    margin: 0;
    font-size: var(--text-xl);
    font-weight: 700;
    letter-spacing: -0.01em;
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
}

/* ── Búsqueda (Inicio) ───────────────────────────────────────────────── */
.search-bar {
    position: relative;
    margin-bottom: var(--space-3);
}
.search-bar__icon {
    position: absolute;
    left: 12px;
    top: 50%;
    transform: translateY(-50%);
    color: var(--fg-faint);
    pointer-events: none;
}
.search-bar__input {
    width: 100%;
    min-height: 44px;
    padding: 0 var(--space-3) 0 40px;
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-btn);
    color: var(--fg);
    font-size: var(--text-base);
    -webkit-appearance: none;
    appearance: none;
}
.search-bar__input:focus-visible {
    outline: none;
    border-color: var(--accent-live);
    box-shadow: 0 0 0 2px color-mix(in srgb, var(--accent-live) 25%, transparent);
}
/* Botón nativo "×" de type=search — dark-friendly. */
.search-bar__input::-webkit-search-cancel-button { filter: invert(0.6); }
.search-bar__vacio {
    color: var(--fg-muted);
    font-size: var(--text-sm);
    text-align: center;
    padding: var(--space-4) 0;
}

/* ── Lista de clientes (Inicio) ─────────────────────────────────────── */
.cliente-list {
    list-style: none;
    margin: 0;
    padding: 0;
    display: flex;
    flex-direction: column;
    gap: var(--space-2);
}
.cliente-row__link {
    display: flex;
    align-items: center;
    justify-content: space-between;
    gap: var(--space-3);
    min-height: 56px;
    padding: var(--space-2) var(--space-3);
    background: var(--bg-card);
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
    color: var(--fg);
    transition: border-color var(--dur-base) var(--ease), background var(--dur-base) var(--ease);
}
.cliente-row__link:hover,
.cliente-row__link:focus-visible {
    border-color: var(--border-hover);
    background: var(--bg-soft);
}
.cliente-row__main {
    display: flex;
    flex-direction: column;
    gap: 2px;
    min-width: 0;
}
.cliente-row__nombre {
    font-weight: 600;
    font-size: var(--text-md);
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
}
.cliente-row__cocheras {
    font-size: var(--text-xs);
    color: var(--fg-muted);
    font-variant-numeric: tabular-nums;
}
.cliente-row__cocheras--sin {
    color: var(--fg-faint);
    font-style: italic;
}

/* ── Estado de saldo — variantes PROPIAS (no err/ok/info del macro
   _tag.html: tag--info ahí es accent NARANJA, y la spec exige AZUL para
   "adelantado"). Mismos 3 estados en el chip de lista (`.tag--*`) y en
   el hero de /m/cliente/{id} (`.saldo-hero--*`). Color + texto SIEMPRE
   juntos (spec: nunca sólo color). ─────────────────────────────────── */
.tag--debe {
    border-color: color-mix(in srgb, var(--err) 45%, var(--border));
    color: var(--err);
}
.tag--aldia {
    border-color: color-mix(in srgb, var(--ok) 45%, var(--border));
    color: var(--ok);
}
.tag--adelantado {
    border-color: color-mix(in srgb, var(--info) 45%, var(--border));
    color: var(--info);
}

/* ── Hero de saldo (/m/cliente/{id}) ────────────────────────────────── */
.saldo-hero {
    display: flex;
    flex-direction: column;
    align-items: center;
    gap: 4px;
    text-align: center;
    padding: var(--space-4) var(--space-3);
    margin-bottom: var(--space-3);
    background: var(--bg-card);
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
}
.saldo-hero__label {
    margin: 0;
    font-size: var(--text-xs);
    font-weight: 600;
    color: var(--fg-muted);
    text-transform: uppercase;
    letter-spacing: 0.06em;
}
.saldo-hero__valor {
    margin: 4px 0 0;
    font-size: 40px;
    font-weight: 700;
    letter-spacing: -0.02em;
    line-height: 1.1;
}
.saldo-hero__sub {
    margin: 2px 0 0;
    font-size: var(--text-sm);
    color: var(--fg-muted);
}
.saldo-hero--debe .saldo-hero__valor { color: var(--err); }
.saldo-hero--aldia .saldo-hero__valor { color: var(--ok); }
.saldo-hero--adelantado .saldo-hero__valor { color: var(--info); }

/* ── Accesos directos (Cargar pago / Cargar gasto) — grandes, spec ─────── */
.cta-row {
    display: flex;
    gap: var(--space-2);
    margin-bottom: var(--space-3);
}
.cta-row__btn {
    flex: 1 1 0;
    justify-content: center;
}

/* ── Secciones tipo card (cocheras/movimientos/mas-list) ────────────── */
.section-block {
    margin-bottom: var(--space-3);
}
.section-block__head {
    display: flex;
    align-items: baseline;
    justify-content: space-between;
    gap: var(--space-2);
    margin-bottom: var(--space-3);
}
.section-block__head .section-title { margin-bottom: 0; }
.section-block__link {
    font-size: var(--text-sm);
    color: var(--accent);
    flex: 0 0 auto;
}
.section-block__vacio {
    color: var(--fg-faint);
    font-size: var(--text-sm);
    margin: 0;
}

.asign-list,
.mov-list,
.mas-list {
    list-style: none;
    margin: 0;
    padding: 0;
    display: flex;
    flex-direction: column;
}
.asign-item,
.mov-item,
.mas-list__item {
    display: flex;
    align-items: center;
    justify-content: space-between;
    gap: var(--space-2);
    padding: var(--space-2) 0;
    min-height: 44px;
}
.asign-item + .asign-item,
.mov-item + .mov-item,
.mas-list__item + .mas-list__item {
    border-top: 1px solid var(--border);
}
.asign-item__codigo {
    font-weight: 600;
    font-variant-numeric: tabular-nums;
}
.asign-item__tarifa,
.mas-list__sub {
    font-size: var(--text-sm);
    color: var(--fg-muted);
}
.mov-item__main,
.mas-list__main {
    display: flex;
    flex-direction: column;
    gap: 2px;
    min-width: 0;
}
.mov-item__concepto,
.mas-list__title {
    font-size: var(--text-sm);
    color: var(--fg);
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
}
.mov-item__fecha {
    font-size: var(--text-xs);
    color: var(--fg-faint);
}
.mov-item__monto,
.mas-list__valor {
    flex: 0 0 auto;
    font-weight: 600;
}
.mov-item__monto.is-positive { color: var(--ok); }
.mov-item__monto.is-negative { color: var(--err); }

/* ── TC del día (Más) ────────────────────────────────────────────────── */
.tc-actual {
    display: flex;
    align-items: baseline;
    gap: var(--space-2);
    margin: 0 0 var(--space-3);
    flex-wrap: wrap;
}
.tc-actual__valor {
    font-size: var(--text-kpi);
    font-weight: 700;
}
.tc-actual__meta {
    font-size: var(--text-xs);
    color: var(--fg-faint);
}

/* ── Secciones secundarias (editar datos / archivar / salir) ───────────
   <details> nativo — cero JS para colapsar. `summary` estilado como fila
   tocable (44px), sin el marker triangular default del browser. ──────── */
.collapse,
.section-secundaria {
    margin-bottom: var(--space-2);
}
.collapse summary,
.section-secundaria summary {
    list-style: none;
    cursor: pointer;
    min-height: 44px;
    display: flex;
    align-items: center;
    gap: var(--space-2);
    padding: var(--space-2) var(--space-3);
    background: var(--bg-card);
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
    color: var(--fg-muted);
    font-size: var(--text-sm);
    font-weight: 600;
    transition: color var(--dur-base) var(--ease), border-color var(--dur-base) var(--ease);
}
.collapse summary::-webkit-details-marker { display: none; }
.collapse summary::before {
    content: "›";
    display: inline-block;
    transform: rotate(0deg);
    transition: transform var(--dur-base) var(--ease);
    color: var(--fg-faint);
}
.collapse[open] summary::before { transform: rotate(90deg); }
.collapse summary:hover,
.collapse summary:focus-visible {
    color: var(--fg);
    border-color: var(--border-hover);
}
.collapse[open] summary {
    border-bottom-left-radius: 0;
    border-bottom-right-radius: 0;
}
.collapse > *:not(summary),
.section-secundaria > *:not(summary) {
    padding: var(--space-3);
    background: var(--bg-card);
    border: 1px solid var(--border);
    border-top: none;
    border-bottom-left-radius: var(--radius-card);
    border-bottom-right-radius: var(--radius-card);
}
.collapse form {
    display: flex;
    flex-direction: column;
    gap: var(--space-3);
}
.section-secundaria--danger summary { color: var(--err); }
.section-secundaria__hint {
    margin: 0 0 var(--space-3);
    font-size: var(--text-sm);
    color: var(--fg-muted);
}
.section-secundaria--salir {
    display: flex;
    align-items: center;
    justify-content: space-between;
    gap: var(--space-3);
    padding: var(--space-2) 0;
}
.section-secundaria--salir form { margin: 0; }

/* ── Forms — `.form-field__input` NO tiene estilo base en el sistema
   (gap conocido del design system, ver comentario en
   services/mrp/app/static/css/style.css:3751). Se estila acá porque los
   forms son el NÚCLEO de esta superficie (3 de 6 pantallas son forms) —
   no tiene sentido scopearlo por página como hace mrp. ────────────────── */
.form-stack {
    display: flex;
    flex-direction: column;
    gap: var(--space-3);
}
.form-field__label {
    display: block;
    margin-bottom: 6px;
    font-size: var(--text-sm);
    font-weight: 600;
    color: var(--fg-muted);
}
.form-field__req {
    color: var(--err);
    margin-left: 2px;
}
.form-field__input {
    width: 100%;
    min-height: 44px;
    padding: 0 var(--space-3);
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-btn);
    color: var(--fg);
    font-family: inherit;
    font-size: var(--text-base);
    -webkit-appearance: none;
    appearance: none;
}
textarea.form-field__input {
    min-height: 0;
    padding: var(--space-2) var(--space-3);
    resize: vertical;
}
select.form-field__input {
    background-image: linear-gradient(45deg, transparent 50%, var(--fg-muted) 50%),
                       linear-gradient(135deg, var(--fg-muted) 50%, transparent 50%);
    background-position: calc(100% - 20px) center, calc(100% - 15px) center;
    background-size: 5px 5px, 5px 5px;
    background-repeat: no-repeat;
    padding-right: 36px;
}
.form-field__input::placeholder { color: var(--fg-faint); }
.form-field__input:hover { border-color: var(--border-hover); }
.form-field__input:focus-visible,
.form-field__input:focus {
    outline: none;
    border-color: var(--accent-live);
    box-shadow: 0 0 0 2px color-mix(in srgb, var(--accent-live) 25%, transparent);
}
.form-field__hint {
    margin: 6px 0 0;
    font-size: var(--text-xs);
    color: var(--fg-faint);
}
.form-field__error {
    margin: 6px 0 0;
    font-size: var(--text-xs);
    color: var(--err);
}
.form-field--collapsed {
    padding-top: var(--space-1);
}
.form-error {
    padding: var(--space-2) var(--space-3);
    background: color-mix(in srgb, var(--err) 12%, var(--bg-card));
    border: 1px solid color-mix(in srgb, var(--err) 45%, var(--border));
    border-radius: var(--radius-card);
    color: var(--err);
    font-size: var(--text-sm);
}

/* Monto — el campo MÁS importante de los forms de carga: grande, alineado
   a la derecha, tabular. inputmode=decimal en el HTML (no type=number: sin
   spinner nativo, acepta coma es-AR). */
.monto-input {
    font-size: 26px;
    font-weight: 700;
    text-align: right;
    font-variant-numeric: tabular-nums;
    letter-spacing: -0.01em;
}

/* ── Chips (método / categoría / moneda) — segmented control táctil ─────
   Grupo de botones tipo pill; el seleccionado toma fondo accent. Nunca
   menos de 44px de alto (WCAG tap target). ──────────────────────────── */
.chip-group {
    display: flex;
    gap: var(--space-2);
    flex-wrap: wrap;
}
.chip-option {
    flex: 1 1 0;
    min-width: 84px;
    min-height: 44px;
    padding: 0 var(--space-2);
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-btn);
    color: var(--fg-muted);
    font-family: inherit;
    font-size: var(--text-sm);
    font-weight: 600;
    cursor: pointer;
    transition: background var(--dur-base) var(--ease), border-color var(--dur-base) var(--ease),
                color var(--dur-base) var(--ease);
}
.chip-option:hover { border-color: var(--border-hover); }
.chip-option.is-active {
    background: var(--accent);
    border-color: var(--accent);
    color: var(--fg-inverse);
}

/* ── Stepper (meses que cubre un pago) ──────────────────────────────── */
.stepper {
    display: flex;
    align-items: stretch;
    gap: var(--space-2);
}
.stepper__btn {
    width: 44px;
    height: 44px;
    flex: 0 0 auto;
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-btn);
    color: var(--fg);
    font-size: var(--text-lg);
    line-height: 1;
    cursor: pointer;
}
.stepper__btn:hover { border-color: var(--border-hover); background: var(--bg-card); }
.stepper__btn:active { transform: scale(0.96); }
.stepper__input {
    width: 64px;
    min-height: 44px;
    text-align: center;
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-btn);
    color: var(--fg);
    font-size: var(--text-md);
    font-weight: 600;
    font-variant-numeric: tabular-nums;
    cursor: default;
}

/* ── Selector de cliente buscable (pago_form) ───────────────────────── */
.cliente-picker__selected {
    display: flex;
    align-items: center;
    justify-content: space-between;
    gap: var(--space-2);
    min-height: 44px;
    padding: 0 var(--space-3);
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-btn);
}
.cliente-picker__nombre {
    font-weight: 600;
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
}
.cliente-picker__resultados {
    list-style: none;
    margin: var(--space-2) 0 0;
    padding: 0;
    max-height: 240px;
    overflow-y: auto;
    -webkit-overflow-scrolling: touch;
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
}
.cliente-picker__item {
    min-height: 44px;
    display: flex;
    align-items: center;
    padding: 0 var(--space-3);
    cursor: pointer;
    font-size: var(--text-sm);
    border-bottom: 1px solid var(--border);
    transition: background var(--dur-base) var(--ease);
}
.cliente-picker__item:last-child { border-bottom: none; }
.cliente-picker__item:hover,
.cliente-picker__item:focus-visible {
    background: var(--bg-soft);
    outline: none;
}
.cliente-picker__vacio {
    margin: var(--space-2) 0 0;
    color: var(--fg-faint);
    font-size: var(--text-sm);
    text-align: center;
}

/* ── Toasts (éxito/error tras un submit) ────────────────────────────── */
.toast-root {
    position: fixed;
    left: var(--space-3);
    right: var(--space-3);
    bottom: max(var(--space-3), var(--safe-area-bottom));
    z-index: 400;
    display: flex;
    flex-direction: column;
    gap: var(--space-2);
    pointer-events: none;
}
.toast {
    padding: var(--space-2) var(--space-3);
    background: var(--surface-elev, var(--bg-card));
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
    color: var(--fg);
    font-size: var(--text-sm);
    box-shadow: var(--shadow-pop);
    opacity: 0;
    transform: translateY(8px);
    transition: opacity var(--dur-slow) var(--ease-out), transform var(--dur-slow) var(--ease-out);
}
.toast.is-visible {
    opacity: 1;
    transform: translateY(0);
}
.toast--err { border-color: color-mix(in srgb, var(--err) 45%, var(--border)); color: var(--err); }
.toast--ok { border-color: color-mix(in srgb, var(--ok) 45%, var(--border)); }

/* Spinner chico para botones en estado "Guardando…" — sólo esta 1 excepción
   de position:fixed-adjacent (toast-root) en toda la superficie mobile; el
   resto del chrome respeta "nunca fixed" (nav, banner) por los gotchas de
   iOS standalone documentados arriba. El toast SÍ necesita overlay real. */
.btn-spinner {
    display: inline-block;
    width: 14px;
    height: 14px;
    border: 2px solid color-mix(in srgb, currentColor 30%, transparent);
    border-top-color: currentColor;
    border-radius: 50%;
    animation: btn-spinner-rotar 700ms linear infinite;
}
@keyframes btn-spinner-rotar {
    to { transform: rotate(360deg); }
}
@media (prefers-reduced-motion: reduce) {
    .btn-spinner { animation-duration: 1400ms; }
}


/* ════════════════════════════════════════════════════════════════════
   Ola D — desktop: superficie completa (/d/*). Tablas, KPIs, modales,
   reportes (echarts) y la grilla de planilla. Todo lo de acá abajo aplica
   SOLO a páginas que extienden desktop/base_d.html (el chrome del sidebar
   ya se extendió arriba, en el bloque "Desktop: sidebar" de la ola A/D).
   ════════════════════════════════════════════════════════════════════ */

/* ── Fondo Vanta NET — SOLO desktop (diseno.md § Estética). Mismo patrón
   visual que reports/mes/meetings: textura CSS estática de fallback + fade a
   canvas animado cuando el vendor carga (gate is-scrolling en static/js/d/
   app.js). Sin scoping bajo .layout-d: el id #vanta-bg sólo existe en el DOM
   de base_d.html (mobile no lo monta), así que esta regla es inerte ahí. ── */
#vanta-bg {
    position: fixed;
    inset: 0;
    width: 100%;
    height: 100%;
    z-index: -1;
    pointer-events: none;
    display: block;
    background-color: var(--bg);
    background-image:
        radial-gradient(130% 90% at 50% -20%, rgba(242, 169, 0, 0.07), rgba(242, 169, 0, 0) 55%),
        radial-gradient(rgba(242, 169, 0, 0.05) 1px, transparent 1.6px);
    background-size: auto, 24px 24px;
    background-position: center top, center;
}
#vanta-bg > canvas {
    pointer-events: none !important;
    opacity: 0;
    transition: opacity 900ms var(--ease-out, ease);
}
#vanta-bg.vanta-ready > canvas { opacity: 1; }
#vanta-bg.vanta-bg--net { opacity: 0.5; }
body.layout-d { background: transparent; }
@media print {
    #vanta-bg { display: none; }
}

/* ── page-toolbar: título + acciones/selector de mes de cada página /d/*. */
.page-toolbar {
    display: flex;
    align-items: center;
    gap: var(--space-3);
    margin-bottom: var(--space-4);
    flex-wrap: wrap;
}
.page-toolbar__title {
    margin: 0;
    font-size: var(--text-xl);
    font-weight: 700;
    letter-spacing: -0.01em;
    flex: 1 1 auto;
}
.page-toolbar .page-head__back {
    display: inline-flex;
    align-items: center;
    justify-content: center;
    width: 36px;
    height: 36px;
    flex: 0 0 auto;
    border-radius: var(--radius-btn);
    color: var(--fg-muted);
    transition: color var(--dur-base) var(--ease), background var(--dur-base) var(--ease);
}
.page-toolbar .page-head__back:hover {
    color: var(--fg);
    background: var(--bg-soft);
}

/* ── mes-selector: ‹ AAAA-MM › — /d/pagos, /d/gastos, /d/reportes,
   /d/planilla. */
.mes-selector {
    display: flex;
    align-items: center;
    gap: var(--space-2);
    padding: 2px;
    background: var(--bg-soft);
    border: 1px solid var(--border);
    border-radius: var(--radius-btn);
}
.mes-selector__btn {
    display: inline-flex;
    align-items: center;
    justify-content: center;
    width: 28px;
    height: 28px;
    border-radius: var(--radius-chip);
    color: var(--fg-muted);
    text-decoration: none;
    font-size: var(--text-lg);
    line-height: 1;
    transition: background var(--dur-base) var(--ease), color var(--dur-base) var(--ease);
}
.mes-selector__btn:hover {
    background: var(--bg-card);
    color: var(--fg);
}
.mes-selector__label {
    min-width: 84px;
    text-align: center;
    font-weight: 600;
    font-size: var(--text-sm);
}

/* ── KPI row (/d, cockpit de 4 tarjetas) ─────────────────────────────── */
.kpi-row {
    display: grid;
    grid-template-columns: repeat(auto-fit, minmax(220px, 1fr));
    gap: var(--space-3);
    margin-bottom: var(--space-3);
}

/* ── accesos rápidos (/d) — botones NO estirados a diferencia del cta-row
   mobile (acá hay espacio de sobra, forzar flex:1 se vería raro). ──────── */
.cta-row-d {
    display: flex;
    gap: var(--space-2);
    margin-bottom: var(--space-4);
}

/* ── toolbar de filtros (/d/clientes: buscador + chip de estado) ───────── */
.toolbar-filtros {
    display: flex;
    align-items: center;
    justify-content: space-between;
    gap: var(--space-3);
    margin-bottom: var(--space-3);
    flex-wrap: wrap;
}
.search-bar--d {
    flex: 1 1 280px;
    max-width: 360px;
    margin-bottom: 0;
}

/* ── acciones dentro de una celda de tabla (Editar/Anular/Cambiar tarifa) */
.tbl-acciones {
    display: flex;
    gap: var(--space-2);
    flex-wrap: wrap;
}

/* ── historial de versiones de tarifa (fila anidada, toggle JS) ────────── */
.tbl-historial-row td {
    background: var(--bg-soft);
    padding: var(--space-2) var(--space-3);
}
.tbl--nested {
    width: auto;
    margin: 0;
}
.tbl--nested th,
.tbl--nested td {
    padding: 4px var(--space-2);
    font-size: var(--text-xs);
}

/* ── utilidades de color (ledger/saldo en tablas desktop) ───────────────
   Mismos tokens que .mov-item__monto.is-positive/-negative (mobile), acá
   genéricos porque se reusan en varias tablas (ledger completo, planilla). */
.is-positive { color: var(--ok); }
.is-negative { color: var(--err); }
.fg-faint { color: var(--fg-faint); }

/* ── /d/clientes/{id} — layout 2 columnas (contenido principal + panel
   lateral de edición/archivar) ──────────────────────────────────────────── */
.detalle-grid {
    display: grid;
    grid-template-columns: minmax(0, 2fr) minmax(260px, 1fr);
    gap: var(--space-4);
    align-items: start;
}
.detalle-col-main,
.detalle-col-side {
    display: flex;
    flex-direction: column;
    gap: var(--space-4);
}
.saldo-hero-acciones {
    display: flex;
    gap: var(--space-2);
    margin-top: var(--space-3);
}
@media (max-width: 1100px) {
    .detalle-grid { grid-template-columns: 1fr; }
}

/* ── /d/clientes/nuevo — form 2 columnas ────────────────────────────────── */
.form-cols {
    display: grid;
    grid-template-columns: 1fr 1fr;
    gap: var(--space-4) var(--space-5);
    max-width: 900px;
}
.form-col {
    display: flex;
    flex-direction: column;
    gap: var(--space-3);
}
.form-col__title {
    margin: 0;
    font-size: var(--text-sm);
    font-weight: 700;
    color: var(--fg-muted);
    text-transform: uppercase;
    letter-spacing: 0.04em;
}
.form-error--full,
.form-actions--full {
    grid-column: 1 / -1;
}
@media (max-width: 760px) {
    .form-cols { grid-template-columns: 1fr; }
}

/* Alta con muchas cocheras: grupos (una tarifa + multi-select de cocheras) */
.grupos-list {
    display: flex;
    flex-direction: column;
    gap: var(--space-3);
}
.grupo-card {
    display: flex;
    flex-direction: column;
    gap: var(--space-3);
    padding: var(--space-3);
    background: color-mix(in srgb, var(--bg-card) 55%, transparent);
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
}
.grupo-card__head {
    display: flex;
    align-items: flex-start;
    gap: var(--space-2);
}
.grupo-card__tarifa { flex: 1 1 auto; min-width: 0; }
.grupo-card__remove {
    flex: 0 0 auto;
    margin-top: 26px; /* baja el botón hasta la altura del <select>, bajo su label */
    display: inline-flex;
    align-items: center;
    justify-content: center;
    width: 34px;
    height: 34px;
    padding: 0;
    color: var(--fg-muted);
    background: transparent;
    border: 1px solid var(--border);
    border-radius: var(--radius-btn, 10px);
    cursor: pointer;
    transition: color .15s, border-color .15s, background .15s;
}
.grupo-card__remove:hover {
    color: var(--err);
    border-color: color-mix(in srgb, var(--err) 45%, var(--border));
    background: color-mix(in srgb, var(--err) 10%, transparent);
}
.grupo-card__remove .icon,
.grupos-add .icon { width: 16px; height: 16px; }
.grupos-add { align-self: flex-start; }

/* Multi-select de cocheras (checklist con búsqueda) */
.cochera-picker { display: flex; flex-direction: column; gap: var(--space-2); }
.cochera-picker__toolbar {
    display: flex;
    align-items: center;
    gap: var(--space-2);
    flex-wrap: wrap;
}
.cochera-picker__search { flex: 1 1 140px; min-width: 0; }
.btn.btn--sm {
    padding: 6px 10px;
    font-size: var(--text-xs);
}
.cochera-picker__count {
    margin-left: auto;
    font-size: var(--text-xs);
    font-weight: 600;
    color: var(--fg-muted);
    white-space: nowrap;
}
.cochera-checklist {
    display: grid;
    grid-template-columns: repeat(auto-fill, minmax(120px, 1fr));
    gap: var(--space-1);
    max-height: 240px;
    overflow-y: auto;
    padding: var(--space-2);
    background: var(--bg-input, color-mix(in srgb, var(--bg-card) 80%, #000 6%));
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
}
.cochera-check {
    display: flex;
    align-items: baseline;
    gap: 8px;
    padding: 6px 8px;
    border-radius: var(--radius-btn, 10px);
    cursor: pointer;
    line-height: 1.2;
    transition: background .12s;
}
.cochera-check:hover { background: color-mix(in srgb, var(--accent-live) 10%, transparent); }
.cochera-check__box { flex: 0 0 auto; accent-color: var(--accent-live); cursor: pointer; }
.cochera-check__code { font-weight: 600; color: var(--fg); }
.cochera-check__loc {
    font-size: var(--text-xs);
    color: var(--fg-faint);
    overflow: hidden;
    text-overflow: ellipsis;
    white-space: nowrap;
}
.cochera-check--off { opacity: .4; cursor: not-allowed; }
.cochera-check--off:hover { background: transparent; }
.cochera-check--off .cochera-check__box { cursor: not-allowed; }
.cochera-picker__empty {
    margin: 0;
    padding: var(--space-2);
    font-size: var(--text-xs);
    color: var(--fg-faint);
    text-align: center;
}

/* ── /d/reportes — P&L de 2 columnas + charts ────────────────────────────── */
.reportes-tc-nota {
    margin: 0 0 var(--space-4);
    font-size: var(--text-sm);
    color: var(--fg-muted);
}
.reportes-tc-nota--falta {
    color: var(--err);
}
.pnl-grid {
    display: grid;
    grid-template-columns: 1fr 1fr;
    gap: var(--space-3);
    margin-bottom: var(--space-4);
}
.pnl-card {
    padding: var(--space-4);
    background: var(--bg-card);
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
}
.pnl-card__ganancia {
    margin: var(--space-2) 0 var(--space-3);
    font-size: 32px;
    font-weight: 700;
    letter-spacing: -0.02em;
}
.pnl-card__filas {
    margin: 0;
    display: flex;
    flex-direction: column;
    gap: var(--space-2);
}
.pnl-card__fila {
    display: flex;
    align-items: baseline;
    justify-content: space-between;
    gap: var(--space-2);
    padding-top: var(--space-2);
    border-top: 1px solid var(--border);
}
.pnl-card__fila dt {
    color: var(--fg-muted);
    font-size: var(--text-sm);
}
.pnl-card__fila dd {
    margin: 0;
    text-align: right;
}
@media (max-width: 900px) {
    .pnl-grid { grid-template-columns: 1fr; }
}

/* Contenedor de chart — tamaño fijo para que echarts.init() mida algo real
   (sin esto el chart no tiene alto y no se ve). Espejo de reports style.css
   `.chart`. */
.chart-wrap {
    margin-bottom: var(--space-2);
}
.chart {
    width: 100%;
    height: 320px;
    border-radius: var(--radius-card);
}
@keyframes chart-reveal { from { opacity: 0; } to { opacity: 1; } }
.chart.chart-revealed { animation: chart-reveal 320ms var(--ease-out, ease) both; }

/* ── modal genérico (reusa el patrón de reports: backdrop + panel + header +
   body). data-modal-open/data-modal-close cableados en static/js/d/app.js. ── */
.modal {
    position: fixed;
    inset: 0;
    z-index: 500;
    display: flex;
    align-items: center;
    justify-content: center;
}
.modal[hidden] { display: none; }
.modal-backdrop {
    position: absolute;
    inset: 0;
    background: rgba(0, 0, 0, 0.65);
    backdrop-filter: blur(2px);
}
.modal-panel {
    position: relative;
    background: var(--bg-card);
    border: 1px solid var(--border);
    border-radius: var(--radius-card);
    width: min(560px, 92vw);
    max-height: 85vh;
    display: flex;
    flex-direction: column;
    box-shadow: 0 16px 60px rgba(0, 0, 0, 0.6);
}
.modal-header {
    display: flex;
    align-items: center;
    justify-content: space-between;
    padding: var(--space-3);
    border-bottom: 1px solid var(--border);
}
.modal-header h3 {
    margin: 0;
    font-size: var(--text-base);
    font-weight: 600;
}
.modal-close {
    background: transparent;
    border: none;
    color: var(--fg-muted);
    font-size: var(--text-xl);
    cursor: pointer;
    line-height: 1;
}
.modal-close:hover { color: var(--fg); }
.modal-body {
    overflow-y: auto;
    padding: var(--space-3);
}
.modal-body .form-stack { gap: var(--space-3); }

/* ── /d/planilla — grilla tipo planilla ─────────────────────────────────
   Núcleo extraído de services/mrp/app/static/js/plan-grid.js (ver
   static/js/d/planilla-grid.js): mismas clases .plan-cell* que MRP,
   reusadas casi 1:1 (selección/foco/guardando/saved), adaptadas al dominio
   de pagos (celda "agregar" .plan-cell--add en vez de pin/revert). ────── */
.planilla-hint {
    margin: 0 0 var(--space-3);
    font-size: var(--text-xs);
    color: var(--fg-faint);
}
.plan-tabla-wrap {
    max-height: calc(100vh - 220px);
}
.plan-th-cliente,
.plan-fila__cliente {
    position: sticky;
    left: 0;
    z-index: 2;
    background: var(--bg-card);
}
.plan-tabla.tbl--zebra tbody tr:nth-child(even) .plan-fila__cliente {
    background: var(--bg-soft);
}
.plan-cell {
    cursor: pointer;
    font-variant-numeric: tabular-nums;
    text-align: right;
    position: relative;
    transition: background var(--dur-base) var(--ease);
}
.plan-cell:hover {
    background: color-mix(in srgb, var(--accent) 8%, transparent);
}
.plan-cell:focus-visible {
    background: color-mix(in srgb, var(--accent) 8%, transparent);
    outline: 2px solid var(--accent);
    outline-offset: -2px;
}
.plan-cell--add {
    color: var(--fg-faint);
}
.plan-cell--add::after {
    content: "+";
    color: var(--fg-faint);
}
.plan-cell--add:hover::after {
    color: var(--accent);
}
.plan-cell-pad {
    background: transparent;
}
.plan-cell--sel {
    background: color-mix(in srgb, var(--accent) 13%, transparent);
}
.plan-cell--focus {
    box-shadow: inset 0 0 0 2px var(--accent);
    background: color-mix(in srgb, var(--accent) 20%, transparent);
}
.plan-cell__edit {
    width: 100%;
    box-sizing: border-box;
    border: 2px solid var(--accent);
    border-radius: 4px;
    background: var(--bg-card);
    color: var(--fg);
    font: inherit;
    font-variant-numeric: tabular-nums;
    text-align: right;
    padding: 1px 4px;
}
.plan-cell--guardando {
    cursor: progress;
    color: transparent;
}
.plan-cell--guardando::before {
    content: "";
    position: absolute;
    inset: 0;
    margin: auto;
    width: 12px;
    height: 12px;
    border: 2px solid var(--border);
    border-top-color: var(--accent);
    border-radius: 50%;
    animation: cocheras-spin 0.7s linear infinite;
}
@keyframes cocheras-spin {
    to { transform: rotate(360deg); }
}
.plan-cell--saved { animation: planCellSaved 1.1s ease-out; }
@keyframes planCellSaved {
    0%   { background: color-mix(in srgb, var(--ok) 48%, transparent); }
    100% { background: transparent; }
}
@media (prefers-reduced-motion: reduce) {
    .plan-cell--saved {
        animation: none;
        background: color-mix(in srgb, var(--ok) 22%, transparent);
    }
    .plan-cell--guardando::before { animation-duration: 1.4s; }
}

/* ═══ Overrides de cascada sobre polish.css (Capa A, NO se edita) ════════
   QA visual 1440px (wake-up 6). polish.css colapsa .sidebar a un rail de
   76px solo-íconos (diseño de reports: pocas secciones muy conocidas).
   Cocheras tiene 8 destinos con íconos genéricos (file/folder/trending) —
   sin label es adivinanza. `body.layout-d ...` le gana a `.sidebar ...` por
   especificidad sin importar que polish.css cargue después de este archivo
   (la tensión de orden de <link> documentada en la cabecera). */
@media (min-width: 1025px) {
    body.layout-d { --sidebar-w: 208px; }
    body.layout-d .sidebar .brand { display: block; }
    body.layout-d .sidebar nav ul { align-items: stretch; padding: 8px 10px; gap: 2px; }
    body.layout-d .sidebar nav li { width: auto; }
    body.layout-d .sidebar nav a {
        width: auto; height: 42px;
        justify-content: flex-start; gap: 10px;
        padding: 0 12px;
        font-size: 13.5px; line-height: 1;
        border-radius: 10px;
    }
}

/* Tablas desktop: headers a la izquierda como los valores (el default de la
   capa compartida los centraba y la tabla de deudores quedaba "flotando");
   columnas numéricas SIEMPRE a la derecha, header incluido. */
body.layout-d .tbl-base th { text-align: left; }
body.layout-d .tbl-base th.numeric,
body.layout-d .tbl-base td.numeric { text-align: right; }

/* Hero de saldo del detalle de cliente: número protagonista (como el hero
   mobile) y color por estado — color + texto, nunca sólo color. */
body.layout-d .saldo-hero-d .kpi-hero__valor {
    font-size: clamp(2.2rem, 3.2vw, 3rem);
    letter-spacing: -0.01em;
}
body.layout-d .saldo-hero-d--debe .kpi-hero__valor { color: var(--err); }
body.layout-d .saldo-hero-d--aldia .kpi-hero__valor { color: var(--ok); }
body.layout-d .saldo-hero-d--adelantado .kpi-hero__valor { color: var(--info); }

/* Celda de pago cruzado (otra moneda, RO — fuera de la matriz de teclado) */
body.layout-d .plan-cell-ro { color: var(--fg-muted); white-space: nowrap; }

/* ── CRM-specific (services/crm) — agregados sobre la base clonada ─────── */

/* Meta-línea del page-toolbar (subtítulo suave bajo el H1). */
.page-toolbar__meta { margin: 2px 0 0; font-size: 13px; color: var(--fg-faint, #9aa0a6); }

/* Grilla 2 columnas para pares de charts/paneles; colapsa en pantallas angostas. */
.grid-2col { display: grid; grid-template-columns: 1fr 1fr; gap: var(--sp-4, 16px); }
@media (max-width: 980px) { .grid-2col { grid-template-columns: 1fr; } }

/* Celda de texto libre truncada (detalle de gestión en tablas). */
.celda-detalle {
  max-width: 420px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
  color: var(--fg-muted, #b7bcc2);
}

/* Sub-marca del sidebar ("CRM Probusol"). */
.brand__sub { font-weight: 400; opacity: .55; font-size: .8em; margin-left: 2px; }

/* Tablas de datos — port del .tbl canónico de reports (cocheras no lo
   estilizaba: sus tablas eran secundarias; acá son protagonistas). */
.tbl-wrap { overflow-x: auto; }
.tbl {
  width: 100%;
  border-collapse: collapse;
  background: var(--bg-card, rgba(255,255,255,.03));
  border: 1px solid var(--border, rgba(255,255,255,.08));
  border-radius: var(--radius-card, 12px);
  overflow: hidden;
  font-size: var(--text-sm, 13.5px);
}
.tbl th, .tbl td {
  padding: 10px 14px;
  border-bottom: 1px solid var(--border, rgba(255,255,255,.08));
  text-align: left;
}
.tbl th {
  background: var(--bg-soft, rgba(255,255,255,.04));
  color: var(--fg-muted, #b7bcc2);
  font-weight: 500;
  font-size: var(--text-xs, 11.5px);
  text-transform: uppercase;
  letter-spacing: 0.04em;
  white-space: nowrap;
}
.tbl tr:last-child td { border-bottom: none; }
.tbl tbody tr:hover td { background: rgba(255,255,255,.015); }
.tbl--zebra tbody tr:nth-child(odd) td { background: rgba(255,255,255,.012); }
.tbl--zebra tbody tr:hover td { background: rgba(255,255,255,.03); }
.tbl--dense th, .tbl--dense td { padding: 6px 10px; }
.tbl th.numeric, .tbl td.numeric {
  text-align: right;
  font-variant-numeric: tabular-nums;
}
.tbl caption {
  caption-side: top; text-align: left; padding: 0 0 8px;
  color: var(--fg-muted, #b7bcc2); font-size: var(--text-xs, 11.5px);
}

/* ── /vendedoras — filtro desde/hasta ────────────────────────────────────
   Inputs de fecha lado a lado dentro del .toolbar-filtros genérico (ese
   ya trae flex+wrap); acá sólo se acota el ancho de cada form-field para
   que no ocupe el 100% (default de _form_field.html es fill). */
.filtro-fecha__campos { display: flex; gap: var(--space-3, 12px); flex-wrap: wrap; }
.filtro-fecha__campos .form-field { width: auto; min-width: 160px; margin-bottom: 0; }
.filtro-fecha__acciones { display: flex; gap: var(--space-2, 8px); align-items: flex-end; }

/* ── /vendedoras — grilla de tarjetas por vendedora ──────────────────────
   Cada tarjeta usa la superficie genérica `.card`; acá sólo el layout
   interno (header con momentum, 3 stats, mix de canal como tags). */
.vendedoras-grid {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(260px, 1fr));
  gap: var(--space-3, 12px);
}
.vendedora-card { display: flex; flex-direction: column; gap: var(--space-3, 12px); }
.vendedora-card__head {
  display: flex; align-items: baseline; justify-content: space-between; gap: var(--space-2, 8px);
}
.vendedora-card__nombre { margin: 0; font-size: var(--text-lg, 16px); font-weight: 600; }
.vendedora-card__momentum { font-size: var(--text-sm, 13px); white-space: nowrap; }
.vendedora-card__stats {
  display: grid; grid-template-columns: repeat(3, 1fr); gap: var(--space-2, 8px);
  padding: var(--space-2, 8px) 0; border-top: 1px solid var(--border, rgba(255,255,255,.08));
  border-bottom: 1px solid var(--border, rgba(255,255,255,.08));
}
.vendedora-card__stat { text-align: center; }
.vendedora-card__stat-valor { margin: 0; font-size: var(--text-xl, 20px); font-weight: 600; }
.vendedora-card__stat-label {
  margin: 2px 0 0; font-size: var(--text-xs, 11.5px); color: var(--fg-faint, #9aa0a6);
  text-transform: uppercase; letter-spacing: 0.03em;
}
.vendedora-card__momentum-detalle { margin: 0; font-size: var(--text-xs, 11.5px); }
.vendedora-card__canales { display: flex; gap: var(--space-2, 8px); flex-wrap: wrap; }

/* ── /gestiones — fila de filtros (form GET) ─────────────────────────────
   Grid responsive de .form-field (usan `.form-field__input`, con estilo
   base ya definido arriba en esta misma hoja). `__q` ocupa 2 columnas para
   que el buscador de texto libre tenga más aire; colapsa a 1 en mobile. */
.gestiones-filtros {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
  gap: var(--space-3, 12px);
  align-items: end;
}
.gestiones-filtros .form-field { margin-bottom: 0; }
.gestiones-filtros__q { grid-column: span 2; }
.gestiones-filtros__acciones { display: flex; gap: var(--space-2, 8px); }
@media (max-width: 640px) {
  .gestiones-filtros__q { grid-column: span 1; }
}

/* ── /clientes — buscador (form GET) ─────────────────────────────────────
   Campo de búsqueda ancho + acciones a la derecha; colapsa en mobile. */
.clientes-buscador {
  display: flex;
  align-items: flex-end;
  gap: var(--space-3, 12px);
  flex-wrap: wrap;
}
.clientes-buscador__campo { flex: 1 1 320px; margin-bottom: 0; }
.clientes-buscador__acciones { display: flex; gap: var(--space-2, 8px); }
/* Sub-línea "hace X" bajo la fecha en la tabla de cartera. */
.clientes-sub { display: block; font-size: var(--text-xs, 11.5px); margin-top: 1px; }
/* Saldo con deuda → rojo (mismo token que el hero de cuenta corriente). */
.clientes-saldo-debe { color: var(--err); }

/* ── /clientes/{cod} — Cliente 360 ───────────────────────────────────────
   Header: identidad a la izquierda, botón Volver a la derecha. */
.cliente-head__ident { flex: 1 1 auto; min-width: 0; }
/* Chips de contexto (canal / zona / grupo) debajo del título. */
.cliente-tags {
  display: flex;
  flex-wrap: wrap;
  gap: var(--space-2, 8px);
  margin: calc(-1 * var(--space-2, 8px)) 0 var(--space-4, 16px);
}
/* Nota chica bajo un section-title (leyenda del gráfico, aclaraciones). */
.section-nota {
  margin: 2px 0 var(--space-3, 12px);
  font-size: var(--text-xs, 11.5px);
  color: var(--fg-faint, #9aa0a6);
}
/* Detalle de gestión SIN truncar — respeta saltos de línea (historia comercial). */
.timeline-detalle {
  white-space: pre-wrap;
  max-width: 640px;
  color: var(--fg-muted, #b7bcc2);
  line-height: 1.45;
}
/* Descripción de artículo pegada a su código (top artículos / pedidos). */
.cliente-art-desc { display: block; font-size: var(--text-xs, 12px); margin-top: 1px; }
/* Utilitario de color faltante (estado de comprobante, textos secundarios). */
.fg-muted { color: var(--fg-muted, #b7bcc2); }
/* En pantallas angostas el detalle no compite por ancho. */
@media (max-width: 720px) { .timeline-detalle { max-width: none; } }

/* ── /pipeline — fila de filtros (form GET) ──────────────────────────────
   Mismo patrón que .gestiones-filtros (grid responsive de .form-field);
   clase propia para no acoplar el nombre a otra página. */
.pipeline-filtros {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
  gap: var(--space-3, 12px);
  align-items: end;
}
.pipeline-filtros .form-field { margin-bottom: 0; }
.pipeline-filtros__q { grid-column: span 2; }
.pipeline-filtros__acciones { display: flex; gap: var(--space-2, 8px); }
@media (max-width: 640px) {
  .pipeline-filtros__q { grid-column: span 1; }
}

/* ── /pipeline — línea secundaria en la celda de razón social
   (nombre de fantasía si difiere, teléfono/email si existen). */
.celda-cliente__sub {
  display: block;
  margin-top: 2px;
  font-size: var(--text-xs, 11.5px);
  color: var(--fg-faint, #9aa0a6);
}

/* ── /ventas — barra de filtros (pills de período + select de canal) ───────
   El período son links GET renderizados como .chip-option en <a> (preservan
   el canal); el canal es un <select> con auto-submit. Fila que envuelve en
   pantallas angostas. Clase propia para no acoplar a otras páginas. */
.ventas-filtros {
  display: flex;
  flex-wrap: wrap;
  gap: var(--space-4, 16px);
  align-items: flex-end;
}
.ventas-filtros__grupo {
  display: flex;
  flex-direction: column;
  gap: var(--space-2, 8px);
  margin: 0;
}
.ventas-filtros__canal {
  display: flex;
  gap: var(--space-2, 8px);
  align-items: center;
}
.ventas-filtros__canal select { min-width: 200px; }
/* Pills inline: compactas (no estiran como el segmented control táctil de los
   forms de carga) y centradas verticalmente al ser <a> con min-height. */
.chip-group--inline { flex-wrap: wrap; }
.chip-group--inline .chip-option {
  flex: 0 0 auto;
  min-width: 56px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  text-decoration: none;
}
/* Tabla de "gaps" — hairline de atención a la izquierda (sutil, no chillón). */
.tbl--gaps { border-left: 2px solid color-mix(in srgb, var(--warn) 55%, transparent); }

/* Línea de datos de alta del prospecto (sección Contactos de la ficha). */
.contacto-alta { margin: 10px 2px 0; font-size: 13px; color: var(--fg-muted, #b7bcc2); }
.contacto-alta a { color: var(--accent); }

/* ── Grupo 360 (/grupos/{nombre}) ─────────────────────────────────────────
   La fila de chips (.cliente-tags) ahora mezcla tags con un btn ghost "Grupo
   → Grupo 360" en la Cliente 360; centrar en el eje cruzado evita que los
   pills se estiren a la altura del botón (align-items default = stretch). */
.cliente-tags { align-items: center; }
/* El link al Grupo 360 desde la ficha individual: compacto, no compite con
   los chips de contexto. */
.cliente-grupo-link { --btn-pad-y: 4px; font-size: var(--text-xs, 12px); }
