/* 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; }
