/* =========================================================
 * Cloud — Reactor
 * Sistema de diseño implementado según DESIGN.md.
 * Archivo único. No fragmentar.
 * ======================================================= */

/* ---------- 1. Tokens ----------
 * Tema único oscuro. Dos zonas cromáticas bien definidas:
 *   1) Sidebar pintado en el rojo institucional #C11313 (color de marca).
 *   2) Resto de la app (topbar, contenido, cards) en grises oscuros neutros.
 * El rojo institucional reaparece como acento (botones primarios, focus,
 * links activos, "ver más", chips activos). No hay modo claro ni toggle.
 */
:root {
    --bg:        #1a1a1a;   /* fondo del área de contenido (gris oscuro) */
    --surface:   #242526;   /* topbar, cards, inputs, dropdowns */
    --border:    #383838;
    --row-hover: #2d2e2f;
    --primary:   #C11313;   /* rojo institucional de marca */
    --primary-h: #8e0e0e;   /* hover más oscuro */
    --danger:    #e62a2a;   /* acciones destructivas */
    --success:   #22c55e;
    --warn:      #f59e0b;
    --info:      #3b82f6;
    --purple:    #8b5cf6;
    --text:      #f0f0f0;
    --muted:     #9ca0a4;
    --radius:    10px;
    --shadow:    0 1px 4px rgba(0,0,0,.45);
    --shadow-lg: 0 8px 32px rgba(0,0,0,.65);
}

/* ---------- 2. Reset & base ---------- */
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }

body {
    font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
    background: var(--bg);
    color: var(--text);
    min-height: 100vh;
}

a { color: inherit; text-decoration: none; }

/* ---------- 3. Layout principal ---------- */
.layout  { display: flex; min-height: 100vh; }

.sidebar {
    width: 220px;
    background: var(--primary);       /* rojo institucional #C11313 */
    border-right: 1px solid rgba(0,0,0,.25);
    display: flex;
    flex-direction: column;
    flex-shrink: 0;
}

.main    { flex: 1; display: flex; flex-direction: column; overflow: hidden; min-width: 0; }

.topbar  {
    background: var(--primary);                /* mismo rojo que el sidebar */
    border-bottom: 1px solid rgba(0,0,0,.25);
    padding: 0 24px;
    height: 60px;
    display: flex;
    align-items: center;
    justify-content: space-between;
    box-shadow: var(--shadow);
    flex-shrink: 0;
}

.content { flex: 1; padding: 24px; overflow-y: auto; }

/* ---------- 4. Sidebar ---------- */
/* El sidebar está pintado en el rojo institucional, por eso sus
 * elementos hijos ignoran los tokens --text/--muted/--border y usan
 * blanco/negro translúcidos: el contraste se calcula contra el rojo,
 * no contra el gris del resto de la app.
 */
.sidebar-logo {
    height: 60px;                /* coincide con la altura de .topbar */
    padding: 0 20px;
    border-bottom: 1px solid rgba(0,0,0,.2);
    display: flex;
    align-items: center;
    justify-content: center;
    flex-shrink: 0;
}
.sidebar-logo-mark  { display: block; width: auto; height: 32px; max-width: 100%; object-fit: contain; }

.sidebar-nav    { padding: 8px 0 12px; flex: 1; }
.sidebar-footer {
    padding: 10px 20px;
    font-size: .75rem;
    color: rgba(255,255,255,.7);
    border-top: 1px solid rgba(0,0,0,.2);
    text-align: center;
    letter-spacing: .03em;
    font-family: monospace;
}

.nav-item {
    display: flex;
    align-items: center;
    gap: 10px;
    padding: 10px 20px;
    font-size: .9rem;
    color: rgba(255,255,255,.85);
    cursor: pointer;
    border-left: 3px solid transparent;
    transition: background .15s, color .15s;
    text-decoration: none;
}
.nav-item:hover  { background: rgba(0,0,0,.18); color: #fff; }
.nav-item.active { background: rgba(0,0,0,.28); color: #fff; border-left-color: #fff; font-weight: 600; }
.nav-icon { font-size: 1.1rem; width: 20px; text-align: center; }

/* Grupos colapsables del sidebar.
 * Vive dentro del rojo institucional, asi que tampoco usa --text/--muted/--border:
 * texto en blanco translucido, bandas internas con negro translucido. */
.nav-group-wrap                       { display: block; }
.nav-group-toggle                     {
    width: 100%;
    background: none;
    border: none;
    text-align: left;
    cursor: pointer;
    font-family: inherit;
    color: rgba(255,255,255,.85);
}
.nav-group-label                      { flex: 1; }
.nav-group-arrow                      {
    margin-left: auto;
    font-size: 1rem;
    font-weight: 700;
    line-height: 1;
    color: rgba(255,255,255,.7);
    transition: transform .2s;
}
.nav-group-wrap.open .nav-group-arrow { transform: rotate(45deg); }

.nav-sub                              {
    display: none;
    background: rgba(0,0,0,.18);
    border-top: 1px solid rgba(0,0,0,.2);
    border-bottom: 1px solid rgba(0,0,0,.2);
}
.nav-group-wrap.open .nav-sub         { display: block; }
.nav-sub-item                         { padding-left: 44px; font-size: .85rem; }
.nav-sub-item.active                  { background: rgba(0,0,0,.32); }

/* ---------- 5. Topbar ----------
 * El topbar comparte el rojo institucional con el sidebar, así que sus
 * hijos siguen la misma regla: blanco / blancos translúcidos para texto
 * y negros translúcidos para estados. No usar --text / --muted / --border.
 */
.topbar-left     { display: flex; align-items: center; gap: 8px; flex: 1; min-width: 0; }
.topbar-title    { font-size: 1rem; font-weight: 600; flex: 1; color: #fff; }

.topbar-user     { display: flex; align-items: center; gap: 8px; flex-shrink: 0; }
.topbar-user-wrap{ position: relative; }
.topbar-username {
    background: none; border: none; cursor: pointer;
    font-size: .85rem; color: rgba(255,255,255,.85);
    display: flex; align-items: center; gap: 6px;
    padding: 6px 10px; border-radius: 8px;
    transition: background .15s, color .15s;
}
.topbar-username:hover { background: rgba(0,0,0,.18); color: #fff; }
.topbar-username i.fa-circle-user { font-size: 1.1rem; }

/* Botones dentro del topbar: ghost adaptado al fondo rojo */
.topbar .btn-ghost       { color: rgba(255,255,255,.85); border-color: rgba(0,0,0,.25); background: transparent; }
.topbar .btn-ghost:hover { background: rgba(0,0,0,.18); color: #fff; }

/* Hamburguesa (mobile) sobre el rojo */
.topbar .hamburger { color: #fff; }

.user-dropdown {
    display: none;
    position: absolute;
    right: 0;
    top: calc(100% + 6px);
    background: var(--surface);
    border: 1px solid var(--border);
    border-radius: 10px;
    box-shadow: var(--shadow-lg);
    min-width: 160px;
    overflow: hidden;
    z-index: 200;
}
.user-dropdown.open { display: block; }
.user-dropdown-item {
    display: block;
    padding: 10px 16px;
    font-size: .85rem;
    color: var(--text);
    cursor: pointer;
    transition: background .15s;
}
.user-dropdown-item:hover { background: var(--bg); color: var(--primary); }

/* ---------- 6. Botones ---------- */
.btn {
    padding: 8px 16px;
    border-radius: var(--radius);
    border: none;
    font-size: .88rem;
    font-weight: 600;
    cursor: pointer;
    display: inline-flex;
    align-items: center;
    gap: 6px;
    transition: background .15s, transform .1s, color .15s, border-color .15s;
    font-family: inherit;
}
.btn:active        { transform: scale(.97); }
.btn-primary       { background: var(--primary); color: #fff; }
.btn-primary:hover { background: var(--primary-h); }
.btn-danger        { background: var(--danger); color: #fff; }
.btn-danger:hover  { background: #c91515; color: #fff; }
.btn-secondary     { background: var(--surface); color: var(--text); border: 1px solid var(--border); }
.btn-secondary:hover { background: var(--bg); }
.btn-ghost         { background: transparent; color: var(--muted); border: 1px solid var(--border); }
.btn-ghost:hover   { background: var(--bg); color: var(--text); }
.btn-sm            { padding: 5px 12px; font-size: .8rem; }
.btn-icon-sm       {
    background: none; border: none; cursor: pointer;
    padding: 4px 8px; border-radius: 6px; font-size: .85rem;
    color: var(--muted);
}
.btn-icon-sm:hover { background: var(--bg); color: var(--text); }

/* ---------- 7. Inputs, selects, textareas ---------- */
input[type=text], input[type=number], input[type=url], input[type=tel],
input[type=email], input[type=date], input[type=time], input[type=datetime-local],
input[type=password], input[type=search], select, textarea {
    border: 1px solid var(--border);
    border-radius: var(--radius);
    padding: 8px 12px;
    font-size: .88rem;
    background: var(--surface);
    color: var(--text);
    outline: none;
    transition: border .15s, box-shadow .15s;
    font-family: inherit;
}
input:focus, select:focus, textarea:focus {
    border-color: var(--primary);
    box-shadow: 0 0 0 3px rgba(193,19,19,.25);
}
input:disabled, select:disabled, textarea:disabled,
input[readonly], textarea[readonly] {
    color: var(--muted); background: var(--bg); cursor: not-allowed; opacity: .75;
}
textarea { resize: vertical; min-height: 60px; }

.field-error   { margin-top: 4px; font-size: .78rem; color: var(--danger); }
.input-invalid { border-color: var(--danger) !important; box-shadow: 0 0 0 2px rgba(239,68,68,.18); }

/* Utilitario: texto secundario / "vacío". Usado en celdas de tabla y
 * dentro de tarjetas de consulta cuando un campo no tiene valor. */
.muted { color: var(--muted); }

/* ---------- 8. Formularios ---------- */
.form-row    { display: grid; grid-template-columns: 1fr 1fr; gap: 12px; }
.form-row-3  { grid-template-columns: repeat(3, 1fr); }
.form-row-4  { grid-template-columns: repeat(4, 1fr); }
.form-group  { display: flex; flex-direction: column; gap: 5px; }
.form-group label { font-size: .8rem; font-weight: 600; color: var(--muted); }
.form-group input,
.form-group select,
.form-group textarea { width: 100%; }

/* ---------- 9. Toolbar ---------- */
.toolbar       { display: flex; align-items: center; gap: 12px; flex-wrap: wrap; margin-bottom: 20px; }
.toolbar-left  { display: flex; align-items: center; gap: 10px; flex: 1; flex-wrap: wrap; }
.toolbar-right { display: flex; gap: 10px; }

.search-wrap   { position: relative; display: inline-flex; align-items: center; }
.search-wrap .search-input { width: 240px; padding-right: 28px; }
.search-clear  {
    position: absolute; right: 6px; background: none; border: none;
    cursor: pointer; color: var(--muted); font-size: 1.1rem;
    padding: 2px 4px; border-radius: 50%; transition: color .15s;
}
.search-clear:hover { color: var(--text); }

.filter-chip   {
    padding: 6px 12px; border-radius: 20px;
    border: 1.5px solid var(--border);
    font-size: .8rem; font-weight: 600; cursor: pointer;
    white-space: nowrap; background: var(--surface);
    color: var(--muted); transition: all .15s;
    font-family: inherit;
}
.filter-chip:hover  { border-color: var(--primary); color: var(--primary); }
.filter-chip.active { background: var(--primary); border-color: var(--primary); color: #fff; }

/* ---------- 10. Tablas ---------- */
.table-card {
    background: var(--surface);
    border: 1px solid var(--border);
    border-radius: var(--radius);
    box-shadow: var(--shadow);
    overflow-x: auto;
    overflow-y: hidden;
}

table       { width: 100%; border-collapse: collapse; font-size: .88rem; }
thead tr    { background: var(--bg); }
th          {
    padding: 10px 14px; text-align: left;
    font-size: .75rem; text-transform: uppercase; letter-spacing: .05em;
    color: var(--muted); font-weight: 600;
    border-bottom: 1px solid var(--border); white-space: nowrap;
}
td          { padding: 10px 14px; border-bottom: 1px solid var(--border); vertical-align: middle; }
tbody tr:last-child td { border-bottom: none; }
tbody tr:hover { background: var(--row-hover); }

.actions     { display: flex; gap: 4px; }
.table-empty { text-align: center; padding: 48px 24px; color: var(--muted); }

.td-img      { width: 44px; height: 44px; object-fit: cover; border-radius: 8px; border: 1px solid var(--border); }
.td-nombre   { font-weight: 600; }
.td-id       { color: var(--muted); font-size: .8rem; font-family: monospace; }

/* ---------- 11. Badges ---------- */
.badge         { display: inline-block; padding: 2px 10px; border-radius: 99px; font-size: .75rem; font-weight: 600; }
.badge-info    { background: rgba(59,130,246,.18);  color: #93c5fd; }
.badge-success { background: rgba(34,197,94,.18);   color: #86efac; }
.badge-danger  { background: rgba(230,42,42,.2);    color: #f5a8a8; }
.badge-warn    { background: rgba(245,158,11,.18);  color: #fcd34d; }

/* ---------- 12. Stat cards ---------- */
.stats-bar  { display: flex; gap: 14px; margin-bottom: 20px; flex-wrap: wrap; }
.stat-card  {
    background: var(--surface);
    border: 1px solid var(--border);
    border-radius: var(--radius);
    padding: 14px 20px;
    display: flex;
    flex-direction: column;
    gap: 2px;
    flex: 1;
    min-width: 120px;
}
.stat-label { font-size: .75rem; color: var(--muted); text-transform: uppercase; letter-spacing: .04em; }
.stat-value { font-size: 1.5rem; font-weight: 700; }
.stat-value.green  { color: var(--success); }
.stat-value.orange { color: var(--primary); }
.stat-value.red    { color: var(--danger); }
.stat-value.muted  { color: var(--muted); }

.dash-link { cursor: pointer; transition: opacity .15s; }
.dash-link:hover { opacity: .75; }
.stat-card.dash-link:hover { background: var(--bg); }

/* ---------- 13. Dashboard grid ---------- */
.dash-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 20px; margin-top: 4px; }
@media (max-width: 768px) { .dash-grid { grid-template-columns: 1fr; } }

.dash-table-header {
    padding: 14px 20px 10px;
    font-weight: 600;
    font-size: .95rem;
    border-bottom: 1px solid var(--border);
    display: flex;
    align-items: center;
    justify-content: space-between;
}
.dash-ver-mas { font-size: .78rem; font-weight: 500; color: var(--primary); }

/* ---------- 13.1 Feed en vivo (dashboard) ---------- */
.dash-live-controls {
    display: flex;
    align-items: center;
    gap: 12px;
}

.dash-live-status {
    display: inline-flex;
    align-items: center;
    gap: 6px;
    font-size: .72rem;
    color: var(--muted);
    text-transform: uppercase;
    letter-spacing: .04em;
    font-weight: 500;
}

.live-dot {
    width: 8px;
    height: 8px;
    border-radius: 50%;
    background: var(--success);
    box-shadow: 0 0 0 0 rgba(34, 197, 94, .6);
    animation: live-pulse 1.4s infinite;
}

.live-paused .live-dot {
    background: var(--muted);
    animation: none;
    box-shadow: none;
}

@keyframes live-pulse {
    0%   { box-shadow: 0 0 0 0   rgba(34, 197, 94, .55); }
    70%  { box-shadow: 0 0 0 8px rgba(34, 197, 94, 0);   }
    100% { box-shadow: 0 0 0 0   rgba(34, 197, 94, 0);   }
}

.live-feed-table tbody tr.is-new {
    animation: live-row-flash 1s ease-out;
}

@keyframes live-row-flash {
    0%   { background-color: rgba(34, 197, 94, .18); }
    100% { background-color: transparent; }
}

.sentido-icon          { font-size: 1rem; }
.sentido-icon.sentido-out { color: var(--success); }   /* S — upload (verde) */
.sentido-icon.sentido-in  { color: var(--info); }      /* E — download (azul) */

/* Mensaje (última columna) en tamaño reducido para no robar peso visual al feed. */
.live-feed-table td:last-child { font-size: .8rem; }

/* ---------- 14. Modales ---------- */
.modal-backdrop {
    position: fixed; inset: 0; background: rgba(0,0,0,.45);
    display: flex; align-items: center; justify-content: center;
    z-index: 100; opacity: 0; pointer-events: none; transition: opacity .2s;
}
.modal-backdrop.open { opacity: 1; pointer-events: all; }
.modal {
    background: var(--surface); border-radius: 14px;
    width: 100%; max-width: 520px; max-height: 90vh; overflow-y: auto;
    box-shadow: var(--shadow-lg);
    transform: scale(.96) translateY(12px); transition: transform .2s;
    margin: 16px;
}
.modal-backdrop.open .modal { transform: scale(1) translateY(0); }
.modal-header   {
    padding: 20px 24px 16px; border-bottom: 1px solid var(--border);
    display: flex; align-items: center; justify-content: space-between;
}
.modal-title    { font-size: 1rem; font-weight: 700; display: flex; align-items: baseline; gap: 10px; flex-wrap: wrap; }
.modal-subtitle { font-size: .8rem; font-weight: 500; color: var(--muted); }
.modal-body     { padding: 20px 24px; display: flex; flex-direction: column; gap: 16px; }
.modal-footer   {
    padding: 16px 24px; border-top: 1px solid var(--border);
    display: flex; gap: 10px; justify-content: flex-end;
}

/* Variante ancha para editores monoespaciados (JSON, logs, etc.). */
.modal.modal-wide { max-width: 760px; }

/* Editor JSON: textarea monoespaciado, altura amplia, no envuelve lineas. */
.json-editor {
    font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, monospace;
    font-size: .82rem;
    line-height: 1.5;
    min-height: 360px;
    max-height: 60vh;
    white-space: pre;
    overflow: auto;
    tab-size: 2;
}

/* ---------- 15. Confirm dialog ---------- */
.confirm-backdrop {
    position: fixed; inset: 0; background: rgba(0,0,0,.5);
    display: flex; align-items: center; justify-content: center;
    z-index: 150; opacity: 0; pointer-events: none; transition: opacity .15s;
}
.confirm-backdrop.open { opacity: 1; pointer-events: all; }
.confirm-box {
    background: var(--surface); border-radius: 14px;
    padding: 28px 28px 20px; max-width: 380px;
    width: calc(100% - 32px); box-shadow: var(--shadow-lg);
    transform: scale(.95); transition: transform .15s;
}
.confirm-backdrop.open .confirm-box { transform: scale(1); }
.confirm-title   { font-weight: 700; margin-bottom: 8px; }
.confirm-msg     { font-size: .88rem; color: var(--muted); margin-bottom: 20px; }
.confirm-actions { display: flex; gap: 10px; justify-content: flex-end; }

/* ---------- 16. Toasts ---------- */
.toast {
    position: fixed; bottom: 24px; left: 50%;
    transform: translateX(-50%) translateY(16px);
    background: #0d0d0d; color: var(--text);
    border: 1px solid var(--border);
    padding: 10px 20px; border-radius: 99px;
    font-size: .88rem; opacity: 0; pointer-events: none;
    transition: opacity .2s, transform .2s;
    z-index: 200; white-space: nowrap;
    box-shadow: var(--shadow-lg);
}
.toast.show  { opacity: 1; transform: translateX(-50%) translateY(0); }
.toast.error { background: var(--danger); }

/* ---------- 17. Toggle switch ---------- */
.toggle-switch { display: flex; align-items: center; gap: 12px; cursor: pointer; user-select: none; }
.toggle-switch input { display: none; }
.toggle-track  {
    width: 44px; height: 24px; border-radius: 12px;
    background: var(--border); position: relative; transition: background .2s;
}
.toggle-switch input:checked + .toggle-track { background: var(--primary); }
.toggle-thumb  {
    position: absolute; top: 3px; left: 3px;
    width: 18px; height: 18px; border-radius: 50%;
    background: #fff; transition: transform .2s;
    box-shadow: 0 1px 4px rgba(0,0,0,.2);
}
.toggle-switch input:checked + .toggle-track .toggle-thumb { transform: translateX(20px); }
.toggle-label { font-size: .9rem; color: var(--text); }

/* ---------- 18. Spinner ---------- */
.spin {
    display: inline-block; width: 28px; height: 28px;
    border: 3px solid var(--border); border-top-color: var(--primary);
    border-radius: 50%; animation: spin .7s linear infinite;
}
@keyframes spin { to { transform: rotate(360deg); } }

/* ---------- 19. Responsive ---------- */
.hamburger {
    display: none; background: none; border: none; cursor: pointer;
    font-size: 1.4rem; color: var(--text);
    padding: 4px 8px; margin-right: 8px; line-height: 1;
}
.sidebar-overlay {
    display: none; position: fixed; inset: 0;
    background: rgba(0,0,0,.45); z-index: 199;
}
.sidebar-overlay.active { display: block; }

@media (max-width: 768px) {
    .hamburger { display: inline-flex; align-items: center; }
    .sidebar   {
        position: fixed; top: 0; left: 0; height: 100vh; z-index: 200;
        transform: translateX(-100%); transition: transform .25s;
    }
    .sidebar.open { transform: translateX(0); }
    .form-row, .form-row-3, .form-row-4 { grid-template-columns: 1fr; }
}

/* ---------- Estado de dispositivo (badges semánticos) ---------- */
.device-status         { display: inline-flex; align-items: center; gap: 6px; }
.device-status::before { content: ""; width: 8px; height: 8px; border-radius: 50%; background: currentColor; }

/* ---------- 21. Menú de acciones (dropdown dentro de modal) ----------
 * Vive sobre el área gris (no sobre el chrome rojo), asi que usa los
 * tokens grises normales --surface / --border / --text / --muted.
 * Pensado para el footer de modales de consulta: trigger a la izquierda
 * (margin-right:auto), "Cerrar" a la derecha. Usar .action-menu-up
 * cuando el dropdown deba abrir hacia arriba para no clipearse.
 */
.action-menu          { position: relative; display: inline-block; }
.action-menu-dropdown {
    display: none;
    position: absolute;
    left: 0;
    top: calc(100% + 6px);
    background: var(--surface);
    border: 1px solid var(--border);
    border-radius: 10px;
    box-shadow: var(--shadow-lg);
    min-width: 220px;
    overflow: hidden;
    z-index: 110;
}
.action-menu.open .action-menu-dropdown { display: block; }
.action-menu-up .action-menu-dropdown   { top: auto; bottom: calc(100% + 6px); }

.action-menu-item {
    display: flex;
    align-items: center;
    gap: 10px;
    width: 100%;
    padding: 10px 16px;
    font-size: .85rem;
    color: var(--text);
    background: none;
    border: none;
    cursor: pointer;
    text-align: left;
    font-family: inherit;
    transition: background .15s, color .15s;
}
.action-menu-item:hover         { background: var(--bg); color: var(--primary); }
.action-menu-item.danger        { color: var(--text); }
.action-menu-item.danger:hover  { background: var(--bg); color: var(--danger); }
.action-menu-item i             { width: 16px; text-align: center; color: var(--muted); }
.action-menu-item:hover i       { color: inherit; }
.action-menu-divider            { height: 1px; background: var(--border); margin: 4px 0; }

/* ---------- 22. Lista de datos (vista de consulta) ----------
 * Pares label/value de solo lectura para modales de "ver detalle".
 * Mismo espaciado y tipografía que las labels de .form-group para
 * que un modal de consulta y uno de edición se vean coherentes.
 */
.data-list  { display: flex; flex-direction: column; gap: 14px; }
.data-row   { display: flex; flex-direction: column; gap: 4px; }
.data-label {
    font-size: .75rem;
    font-weight: 600;
    text-transform: uppercase;
    letter-spacing: .04em;
    color: var(--muted);
}
.data-value {
    font-size: .9rem;
    color: var(--text);
    word-break: break-word;
    white-space: pre-wrap;
}
.data-value.muted { color: var(--muted); font-style: italic; }
.data-value code  {
    font-family: monospace;
    font-size: .85rem;
    background: var(--bg);
    border: 1px solid var(--border);
    border-radius: 6px;
    padding: 2px 8px;
}

/* ---------- 23. ABM: header del módulo ----------
 * Bloque obligatorio al tope de todo listado ABM: título de la entidad
 * en plural + subtítulo descriptivo en una sola frase. Va antes de los
 * KPIs (si los hay) y de la toolbar. Ver ABM.md §1.1.
 */
.module-header   { margin-bottom: 18px; }
.module-title    {
    font-size: 1.35rem;
    font-weight: 700;
    color: var(--text);
    margin: 0 0 4px;
    line-height: 1.2;
}
.module-subtitle {
    font-size: .88rem;
    color: var(--muted);
    margin: 0;
    line-height: 1.4;
}

/* ---------- 24. ABM: columnas de acción del listado ----------
 * Cada ícono (Consultar/Editar/Eliminar) va en su propia columna al
 * final de la tabla, en ese orden. Las columnas son angostas y centradas.
 */
th.action-col, td.action-col {
    width: 1%;
    white-space: nowrap;
    text-align: center;
    padding-left: 4px;
    padding-right: 4px;
}

/* ---------- 25. ABM: tarjetas de consulta (read-only) ----------
 * Modal "Consultar" muestra TODOS los campos del registro. Cada campo es
 * una tarjeta con esquinas redondeadas y fondo exactamente 10% más oscuro
 * que `--surface` (regla obligatoria de ABM.md, no variar por módulo).
 * Tarjetas de 50% para valores cortos (códigos, fechas, estados, booleanos)
 * y 100% para valores largos (descripciones, JSON, payloads, direcciones).
 */
.view-grid {
    display: flex;
    flex-wrap: wrap;
    gap: 12px;
}
.view-card {
    background: color-mix(in srgb, var(--surface) 90%, #000);
    border-radius: var(--radius);
    padding: 12px 14px;
    display: flex;
    flex-direction: column;
    gap: 4px;
    min-width: 0;
}
.view-card-half { flex: 1 1 calc(50% - 6px); }
.view-card-full { flex: 1 1 100%; }
.view-card-label {
    font-size: .75rem;
    font-weight: 600;
    text-transform: uppercase;
    letter-spacing: .04em;
    color: var(--muted);
}
.view-card-value {
    font-size: .9rem;
    color: var(--text);
    word-break: break-word;
    white-space: pre-wrap;
}
.view-card-value.muted { color: var(--muted); font-style: italic; }
.view-card-value code  {
    font-family: monospace;
    font-size: .85rem;
    background: var(--bg);
    border: 1px solid var(--border);
    border-radius: 6px;
    padding: 2px 8px;
}
.view-card-value pre {
    margin: 0;
    font-family: monospace;
    font-size: .82rem;
    background: var(--bg);
    border: 1px solid var(--border);
    border-radius: 6px;
    padding: 10px 12px;
    overflow-x: auto;
    white-space: pre-wrap;
}

@media (max-width: 640px) {
    .view-card-half { flex: 1 1 100%; }
}

/* ---------- 26. ABM: Modal de Filtros ----------
 * Centraliza todos los filtros del listado (ver ABM.md §3). Estructura:
 *   - Header con título "Filtros" + botón ×.
 *   - Body en grilla 2-col (colapsa a 1 col bajo 640px). El primer campo
 *     siempre es `Código`; al final van `Límite`, `Ordenar por`, `Dirección`.
 *   - Footer: Limpiar (ghost) | Cancelar (secondary) | Aplicar (primary),
 *     todos alineados a la derecha por el `.modal-footer` base.
 *
 * Trigger: botón `Filtros` (secondary, icono fa-filter) en la toolbar del
 * listado. La búsqueda rápida del toolbar es sólo un atajo: el modal es
 * la fuente completa de filtros del módulo.
 */
.filters-grid {
    display: grid;
    grid-template-columns: 1fr 1fr;
    gap: 12px 16px;
}
@media (max-width: 640px) {
    .filters-grid { grid-template-columns: 1fr; }
}

/* ---------- 27. Tile grid (menú de navegación) ----------
 * Grilla de tarjetas-botón usadas como menú de aterrizaje hacia
 * sub-pantallas o disparadores de acción (Herramientas, etc.).
 * No es para datos numéricos: para eso está .stat-card (§12).
 */
.tile-grid {
    display: grid;
    grid-template-columns: repeat(auto-fill, minmax(220px, 1fr));
    gap: 16px;
}
.tile-card {
    background: var(--surface);
    border: 1px solid var(--border);
    border-radius: var(--radius);
    padding: 20px;
    display: flex;
    flex-direction: column;
    gap: 6px;
    text-align: left;
    cursor: pointer;
    text-decoration: none;
    color: var(--text);
    font-family: inherit;
    transition: border-color .15s, background .15s, transform .1s;
}
.tile-card:hover  { border-color: var(--primary); background: var(--row-hover); }
.tile-card:active { transform: scale(.98); }
.tile-icon  { font-size: 1.6rem; line-height: 1; margin-bottom: 4px; }
.tile-title { font-weight: 600; font-size: .95rem; color: var(--text); }
.tile-desc  { font-size: .8rem; color: var(--muted); }

/* ---------- 28. Pantalla de login ----------
 * login.php es la unica vista que vive fuera de la SPA y no tiene chrome.
 * Excepcion al patron "rojo solo como acento": el card del login va
 * PINTADO de rojo institucional (#C11313, var(--primary)) — mismo color
 * que el banner del logo de marca, asi la pantalla "es" la marca.
 * Por eso los hijos del card siguen las reglas del chrome rojo (§4-§5):
 *   - tipografia en #fff / opacidades de blanco.
 *   - bordes y rellenos internos en negros translucidos.
 *   - el boton primario invierte la paleta (fondo blanco, texto rojo)
 *     para destacar sobre el fondo rojo sin perder contraste.
 *   - los tokens --text / --muted / --border NO se usan dentro del card.
 */
body.login-page { background: var(--bg); }

.login-shell {
    min-height: 100vh;
    display: flex;
    align-items: center;
    justify-content: center;
    padding: 24px;
}
.login-card {
    width: 100%;
    max-width: 360px;
    background: var(--primary);
    border: 1px solid rgba(0,0,0,.25);
    border-radius: 14px;
    box-shadow: var(--shadow-lg);
    padding: 28px 28px 24px;
    display: flex;
    flex-direction: column;
    gap: 16px;
    color: #fff;
}
.login-brand {
    display: flex;
    align-items: center;
    justify-content: center;
    padding: 4px 0 0;
}
.login-logo     { display: block; height: 44px; width: auto; max-width: 80%; object-fit: contain; }
.login-title    { font-size: 1.2rem; font-weight: 700; text-align: center; color: #fff; margin: 4px 0 0; }
.login-subtitle { font-size: .85rem; color: rgba(255,255,255,.78); text-align: center; margin: -8px 0 0; }

.login-form     { display: flex; flex-direction: column; gap: 14px; }
.login-form .form-group label {
    color: rgba(255,255,255,.88);
}
.login-form input[type=text],
.login-form input[type=password] {
    background: rgba(0,0,0,.22);
    border: 1px solid rgba(0,0,0,.35);
    color: #fff;
}
.login-form input::placeholder { color: rgba(255,255,255,.55); }
.login-form input:focus {
    border-color: #fff;
    box-shadow: 0 0 0 3px rgba(255,255,255,.22);
}
.login-form input.input-invalid {
    border-color: #fff !important;
    box-shadow: 0 0 0 2px rgba(255,255,255,.3);
}

/* Mensaje de error: blanco sobre banda negra translucida (visible sobre rojo) */
.login-error {
    display: block;
    color: #fff;
    background: rgba(0,0,0,.28);
    border: 1px solid rgba(0,0,0,.35);
    border-radius: var(--radius);
    padding: 8px 12px;
    font-size: .82rem;
    font-weight: 500;
}
.login-error[hidden] { display: none; }

/* Boton primario invertido: blanco con texto rojo para contraste sobre rojo */
.login-page .btn-primary {
    background: #fff;
    color: var(--primary);
}
.login-page .btn-primary:hover {
    background: rgba(255,255,255,.9);
    color: var(--primary-h);
}
.login-page .btn-primary:disabled {
    background: rgba(255,255,255,.6);
    color: var(--primary);
    cursor: not-allowed;
}
.login-submit   { justify-content: center; padding: 10px 16px; margin-top: 4px; }
