/*
 * This is a manifest file that'll be compiled into application.css.
 *
 * With Propshaft, assets are served efficiently without preprocessing steps. You can still include
 * application-wide styles in this file, but keep in mind that CSS precedence will follow the standard
 * cascading order, meaning styles declared later in the document or manifest will override earlier ones,
 * depending on specificity.
 *
 * Consider organizing styles into separate files for maintainability.
 */

/* ── Design tokens — rediseño "Atrevida" v3 ────────────────────────────────── */
:root {
  --dp-bg-page:        #fafafa;
  --dp-bg-side-right:  #f5f3ff;
  --dp-sidebar-bg:     rgba(49, 46, 129, 0.82);
  --dp-card-bg:        #ffffff;
  --dp-card-border:    #f1f5f9;

  --dp-brand-500:      #6366f1;
  --dp-brand-600:      #4f46e5;
  --dp-brand-700:      #4338ca;
  --dp-brand-900:      #312e81;
  --dp-violet-500:     #8b5cf6;
  --dp-violet-600:     #7c3aed;
  --dp-fuchsia-500:    #a855f7;

  --dp-text-primary:   #0f172a;
  --dp-text-secondary: #64748b;
  --dp-text-muted:     #94a3b8;

  --dp-cta-gradient:   linear-gradient(135deg, #6366f1 0%, #8b5cf6 50%, #a855f7 100%);
  --dp-cta-shadow:
    0 8px 20px -4px rgba(124, 58, 237, 0.55),
    0 2px 6px rgba(99, 102, 241, 0.30),
    inset 0 1px 0 rgba(255, 255, 255, 0.25);

  --dp-hero-gradient:
    linear-gradient(135deg,
      rgba(79, 70, 229, 0.28) 0%,
      rgba(139, 92, 246, 0.18) 55%,
      rgba(196, 181, 253, 0.10) 100%);
  --dp-hero-border:    1px solid rgba(99, 102, 241, 0.18);
  --dp-hero-shadow:    0 8px 20px -16px rgba(79, 70, 229, 0.20);
}

/* Card shell del rediseño: borde fino, sin sombra, radius 18 */
.dp-card {
  background: var(--dp-card-bg);
  border: 1px solid var(--dp-card-border);
  border-radius: 18px;
  padding: 22px;
  box-shadow: none;
}

/* Barrita decorativa para headers de sección (timeline, trips table, etc.) */
.dp-section-bar {
  width: 4px;
  height: 20px;
  border-radius: 2px;
  background: linear-gradient(180deg, var(--dp-brand-500), var(--dp-violet-500));
  flex-shrink: 0;
}

/* Pill-counter neutra junto a títulos de sección */
.dp-section-pill {
  padding: 2px 10px;
  background: #f5f3ff;
  color: var(--dp-violet-600);
  border-radius: 999px;
  font-size: 11px;
  font-weight: 600;
  white-space: nowrap;
}

/* Hero card del dashboard */
.dp-hero {
  position: relative;
  border-radius: 24px;
  padding: 16px 24px;
  background: var(--dp-hero-gradient);
  border: var(--dp-hero-border);
  box-shadow: var(--dp-hero-shadow);
  color: #1e1b4b;
  /* IMPORTANTE: NO usar overflow:hidden aquí — clipea los dropdowns
     (year selector, country search del setup card). Los orbs decorativos
     viven dentro de un wrapper interior con su propio overflow. */
  overflow: visible;
}
@media (min-width: 640px) {
  .dp-hero { padding: 20px 28px; }
}

/* Wrapper de orbs decorativos — overflow:hidden + radius para contener
   los blurs sin afectar a dropdowns/popovers de los hijos del hero. */
.dp-hero-orbs {
  position: absolute;
  inset: 0;
  border-radius: 24px;
  overflow: hidden;
  pointer-events: none;
  z-index: 0;
}

.dp-hero-eyebrow {
  font-size: 11px;
  letter-spacing: 1.4px;
  font-weight: 600;
  color: #5b21b6;
  margin-bottom: 6px;
}

.dp-hero-greeting {
  font-size: clamp(28px, 4vw, 42px);
  font-weight: 700;
  letter-spacing: -1.2px;
  margin: 0;
  color: #1e1b4b;
}

/* CTA primaria — gradient brand + sombra elevada */
.dp-cta-primary {
  display: inline-flex;
  align-items: center;
  gap: 8px;
  padding: 11px 20px;
  background: var(--dp-cta-gradient);
  color: #fff;
  border: none;
  border-radius: 12px;
  font-size: 14px;
  font-weight: 700;
  letter-spacing: 0.2px;
  cursor: pointer;
  box-shadow: var(--dp-cta-shadow);
  transition: transform 160ms ease, box-shadow 160ms ease;
  white-space: nowrap;
}
.dp-cta-primary:hover {
  transform: translateY(-1px);
  box-shadow:
    0 12px 26px -4px rgba(124, 58, 237, 0.6),
    0 4px 10px rgba(99, 102, 241, 0.35),
    inset 0 1px 0 rgba(255, 255, 255, 0.3);
}

/* Botón secundario glass dentro del hero */
.dp-cta-secondary {
  display: inline-flex;
  align-items: center;
  gap: 6px;
  padding: 10px 16px;
  background: rgba(255, 255, 255, 0.7);
  color: #4c1d95;
  border: 1px solid rgba(124, 58, 237, 0.25);
  border-radius: 12px;
  font-size: 13px;
  font-weight: 600;
  cursor: pointer;
  backdrop-filter: blur(10px);
  -webkit-backdrop-filter: blur(10px);
  transition: background 160ms ease, border-color 160ms ease;
  white-space: nowrap;
}
.dp-cta-secondary:hover {
  background: rgba(255, 255, 255, 0.92);
  border-color: rgba(124, 58, 237, 0.45);
}

/* ── Year selector segmented pill (handoff) ───────────────────────────────── */

/* La pill nunca debe recortar años: envuelve a varias líneas cuando no cabe
   y comprime los segmentos en pantallas estrechas para mostrarlos todos. */
.dp-year-selector {
  flex-wrap: wrap;
  justify-content: center;
}

.dp-year-segment {
  padding: 6px 14px;
  border-radius: 8px;
  font-size: 12px;
  font-weight: 600;
  color: #64748b;
  background: transparent;
  border: none;
  cursor: pointer;
  white-space: nowrap;
  transition: background 160ms ease, color 160ms ease;
  font-variant-numeric: tabular-nums;
}
.dp-year-segment:hover { color: #1e293b; background: rgba(99, 102, 241, 0.06); }
.dp-year-segment.is-active {
  background: linear-gradient(135deg, #6366f1, #8b5cf6);
  color: #ffffff;
  box-shadow: 0 4px 10px -2px rgba(99, 102, 241, 0.4);
}
.dp-year-segment.is-active:hover { color: #ffffff; }

.dp-year-overflow { padding: 6px 8px; color: #94a3b8; }
.dp-year-overflow:hover { color: #4f46e5; }

/* Móvil: comprime los segmentos para que quepan más años por línea. */
@media (max-width: 480px) {
  .dp-year-selector { gap: 2px; }
  .dp-year-segment { padding: 5px 9px; font-size: 11px; }
  .dp-year-overflow { padding: 5px 6px; }
}

/* ── Sidebar compacto izquierdo (lg+) — colapsado por defecto, hover-expand ── */

.dp-compact-sidebar {
  width: 80px;
  transition: width 220ms cubic-bezier(0.22, 1, 0.36, 1);
  /* Forzamos alignment de items al centro cuando colapsado */
  align-items: center;
}
.dp-compact-sidebar:hover,
.dp-sidebar-pinned .dp-compact-sidebar {
  width: 220px;
  align-items: stretch;
  /* Sombra cuando expandido para indicar "está sobre el contenido" */
  box-shadow: 6px 0 24px rgba(15, 23, 42, 0.25);
}

/* Estado pinned: el main se desplaza para hacer hueco al sidebar expandido (lg+).
   Reemplaza el `lg:pl-20` (80px) por 220px sólo cuando hay pin activo. */
@media (min-width: 1024px) {
  .dp-sidebar-pinned main {
    padding-left: 220px;
    transition: padding-left 220ms cubic-bezier(0.22, 1, 0.36, 1);
  }
  main {
    transition: padding-left 220ms cubic-bezier(0.22, 1, 0.36, 1);
  }
}

/* Hamburger: indicar visualmente cuando está activo (pinned) */
.dp-sidebar-pinned .dp-sidebar-hamburger {
  color: #ffffff;
  background: rgba(255, 255, 255, 0.12);
}
/* Alterna iconos hamburger ↔ chevron según estado pinned */
.dp-sidebar-pin-icon { display: none; }
.dp-sidebar-pinned .dp-sidebar-hamburger-icon { display: none; }
.dp-sidebar-pinned .dp-sidebar-pin-icon { display: inline-block; }

/* Brand (icono "D" + wordmark "DayProof") + hamburger */
.dp-sidebar-brand {
  padding: 0 14px;
  width: 100%;
  /* En colapsado: solo el icono D centrado; hamburger oculto */
  justify-content: center;
}
/* Hamburger SIEMPRE visible (es el toggle de pin) */
.dp-sidebar-hamburger { display: flex; }

/* Brand-text (palabra "DayProof") y brand-link (icono "D") solo cuando expandido */
.dp-compact-sidebar:not(:hover) .dp-sidebar-brand-text {
  display: none;
}
.dp-compact-sidebar:not(:hover) .dp-sidebar-brand-link {
  display: none;
}
.dp-sidebar-pinned .dp-compact-sidebar .dp-sidebar-brand-text {
  display: inline;
}
.dp-sidebar-pinned .dp-compact-sidebar .dp-sidebar-brand-link {
  display: flex;
  justify-content: flex-start;
}
.dp-compact-sidebar:hover .dp-sidebar-brand,
.dp-sidebar-pinned .dp-compact-sidebar .dp-sidebar-brand {
  justify-content: flex-start;
}

/* Items: pill cuadrada cuando colapsado, fila horizontal cuando expandido */
.dp-sidebar-item {
  position: relative;
  display: flex;
  align-items: center;
  justify-content: center;
  border-radius: 12px;
  color: #a5b4fc;
  transition: background 160ms ease, color 160ms ease, padding 200ms ease, width 200ms ease;
  width: 44px;
  height: 44px;
  flex-shrink: 0;
  margin: 0 auto;
}
.dp-compact-sidebar:hover .dp-sidebar-item,
.dp-sidebar-pinned .dp-compact-sidebar .dp-sidebar-item {
  width: auto;
  margin: 0;
  justify-content: flex-start;
  padding: 0 12px;
  gap: 12px;
}
.dp-sidebar-icon { display: inline-flex; }

/* Label visible solo cuando el sidebar está expandido */
.dp-sidebar-label {
  font-size: 13px;
  font-weight: 500;
  white-space: nowrap;
  opacity: 0;
  transform: translateX(-6px);
  transition: opacity 140ms ease 60ms, transform 140ms ease 60ms;
  pointer-events: none;
  display: none;
}
.dp-compact-sidebar:hover .dp-sidebar-label,
.dp-sidebar-pinned .dp-compact-sidebar .dp-sidebar-label {
  display: inline;
  opacity: 1;
  transform: translateX(0);
  pointer-events: auto;
}

.dp-sidebar-item:hover {
  background: rgba(255, 255, 255, 0.08);
  color: #ffffff;
}
.dp-sidebar-item.is-active {
  background: rgba(255, 255, 255, 0.10);
  color: #ffffff;
}
.dp-sidebar-item.is-active::before {
  content: "";
  position: absolute;
  left: -2px;
  top: 10px;
  bottom: 10px;
  width: 3px;
  border-radius: 4px;
  background: #a855f7;
}

/* Nav: pasa de items-center a items-stretch al expandir */
.dp-compact-sidebar nav {
  align-items: center;
}
.dp-compact-sidebar:hover nav,
.dp-sidebar-pinned .dp-compact-sidebar nav {
  align-items: stretch;
}

/* Tooltip pill flotante: solo cuando el sidebar está colapsado */
.dp-sidebar-tooltip {
  position: absolute;
  left: calc(100% + 14px);
  top: 50%;
  transform: translateY(-50%) translateX(-4px);
  white-space: nowrap;
  padding: 6px 10px;
  background: #1e1b4b;
  color: #fff;
  font-size: 12px;
  font-weight: 500;
  border-radius: 8px;
  box-shadow: 0 8px 18px rgba(15, 23, 42, 0.25);
  opacity: 0;
  pointer-events: none;
  transition: opacity 140ms ease, transform 140ms ease;
  z-index: 50;
}
.dp-compact-sidebar:not(:hover) .dp-sidebar-item:hover .dp-sidebar-tooltip,
.dp-compact-sidebar:not(:hover) .dp-sidebar-item:focus-visible .dp-sidebar-tooltip {
  opacity: 1;
  transform: translateY(-50%) translateX(0);
}
.dp-compact-sidebar:hover .dp-sidebar-tooltip,
.dp-sidebar-pinned .dp-compact-sidebar .dp-sidebar-tooltip { display: none; }

/* Hide scrollbar in compact sidebar nav */
.scrollbar-none::-webkit-scrollbar { display: none; }
.scrollbar-none { scrollbar-width: none; }

/* Reduced motion: disable expansion animation, keep static collapsed state */
@media (prefers-reduced-motion: reduce) {
  .dp-compact-sidebar,
  .dp-sidebar-item,
  .dp-sidebar-label { transition: none; }
}

/* ── Side panel del dashboard — toggle, colapso y animaciones de actividad ── */

/* Side panel colapsado: ocultarlo + ensanchar la columna principal.
   El controller side-panel añade .side-panel-collapsed al wrapper, que puede ser
   .dashboard-grid (dashboard) o el partial shared/_side_panel_wrapper (trips, tickets, map). */
#dashboard_side_panel {
  transition: width 240ms ease, opacity 200ms ease, padding 200ms ease;
}
.side-panel-collapsed #dashboard_side_panel {
  width: 0 !important;
  min-width: 0 !important;
  padding-left: 0 !important;
  padding-right: 0 !important;
  border-left-width: 0 !important;
  opacity: 0;
  pointer-events: none;
  overflow: hidden;
}

/* Wrapper de side panel: ancla para el botón toggle (xl+).
   En xl+ el wrapper xl:flex actúa como contenedor de posición absoluta. */
.dp-side-panel-wrap { position: relative; }

/* Botón "lengüeta" para mostrar/ocultar el panel lateral (xl+).
   Posicionado en el padding superior del panel (top:14px) — fuera de la zona del
   page_header y del contenido principal. Pegado al borde izquierdo del panel
   cuando está visible; al borde derecho del viewport cuando está colapsado. */
.dp-side-toggle {
  position: absolute;
  top: 14px;
  right: 380px;
  z-index: 30;
  width: 26px;
  height: 32px;
  display: none;
  align-items: center;
  justify-content: center;
  background: #ffffff;
  border: 1px solid rgb(226 232 240);
  border-right: 0;
  border-radius: 8px 0 0 8px;
  color: rgb(99 102 241);
  box-shadow: -3px 2px 10px rgba(15, 23, 42, 0.06);
  cursor: pointer;
  transition: right 240ms cubic-bezier(0.22, 1, 0.36, 1), background 160ms ease, color 160ms ease, box-shadow 160ms ease;
}
@media (min-width: 1280px) {
  .dp-side-toggle { display: inline-flex; }
}
.dp-side-toggle:hover {
  background: rgb(238 242 255);
  color: rgb(79 70 229);
  box-shadow: -3px 2px 14px rgba(99, 102, 241, 0.18);
}
.dp-side-toggle:focus-visible {
  outline: 2px solid rgb(99 102 241);
  outline-offset: 2px;
}
.dp-side-toggle svg {
  width: 14px;
  height: 14px;
  transition: transform 220ms ease;
}
.side-panel-collapsed .dp-side-toggle {
  right: 0;
}
.side-panel-collapsed .dp-side-toggle svg {
  transform: rotate(180deg);
}

/* ── Dashboard search dropdown ────────────────────────────────────────────── */
[data-dashboard-search-target="panel"] {
  opacity: 0;
  transform: translateY(-6px) scale(0.98);
  transition:
    opacity   300ms cubic-bezier(0.16, 1, 0.3, 1),
    transform 300ms cubic-bezier(0.16, 1, 0.3, 1);
}
[data-dashboard-search-target="panel"][data-state="open"] {
  opacity: 1;
  transform: translateY(0) scale(1);
}

/* Animación suave de inserción del nuevo item de actividad */
@keyframes sidePanelActivityFresh {
  0%   { opacity: 0; transform: translateY(-10px); background-color: rgba(165, 180, 252, 0.15); }
  60%  { opacity: 1; transform: translateY(0);    background-color: rgba(165, 180, 252, 0.10); }
  100% { opacity: 1; transform: translateY(0);    background-color: transparent; }
}
.side-panel-activity-fresh {
  animation: sidePanelActivityFresh 800ms cubic-bezier(0.16, 1, 0.3, 1) both;
  border-radius: 8px;
  margin: -2px -4px;
  padding: 2px 4px;
}

/* Item de actividad: transición de movimiento suave cuando uno se inserta arriba
   y los otros se desplazan hacia abajo. */
.side-panel-activity-item {
  transition: transform 280ms cubic-bezier(0.22, 1, 0.36, 1);
}

/* ── Side panel · Card "Billetes analizados" ─────────────────────────────────
   Badge "verified" estilo Material Symbols — el SVG es un burst azul con
   check blanco integrado, así que el wrapper solo aporta hover/transición y
   sombra sutil para dar volumen. */
.dp-side-panel-verified-badge {
  width: 22px;
  height: 22px;
  /* Sin overflow:hidden — el SVG usa overflow:visible para que stroke + halo
     queden completos. Si algún ancestro recorta, este display: block-flex evita
     que line-height añada padding fantasma. */
  display: inline-flex;
  align-items: center;
  justify-content: center;
  transition: transform 180ms cubic-bezier(0.22, 1, 0.36, 1);
}
.dp-side-panel-verified-badge:hover {
  transform: scale(1.08);
}

/* Item de billete del side panel — transición de hover suave. Los items
   ya pintados nunca se animan: los broadcasts incrementales (prepend del
   nuevo <li>) sólo tocan el item entrante, por lo que el reflow del
   resto de la lista no necesita animación explícita — el navegador hace
   el shift cuando el nuevo item crece desde max-height: 0. */
.side-panel-ticket-item {
  transition: background-color 180ms ease;
}

/* Estado vacío del listado: el <p>.side-panel-tickets-empty sólo se
   muestra cuando el <ul#side_panel_recent_tickets_list> está vacío.
   Cuando llega el primer prepend, el <ul> deja de ser `:empty` y el
   mensaje desaparece automáticamente — sin JS ni broadcasts extra. */
.side-panel-tickets-empty { display: none; }
#side_panel_recent_tickets_list:empty + .side-panel-tickets-empty { display: block; }

/* Animación de entrada para items recién insertados via Turbo Stream
   `prepend`. El <li> crece desde `max-height: 0` hasta su altura
   natural empujando suavemente a los items inferiores; en paralelo
   hace fade-in. Duración deliberadamente larga (~1.1s) para que el
   efecto sea claramente perceptible como una animación suave y no se
   confunda con un repaint instantáneo del contenedor. */
@keyframes sidePanelTicketFresh {
  0%   {
    opacity: 0;
    max-height: 0;
    margin-top: 0;
    padding-top: 0;
    padding-bottom: 0;
    transform: translateY(-6px);
    overflow: hidden;
  }
  15%  {
    opacity: 0.05;
  }
  60%  {
    opacity: 0.7;
    transform: translateY(0);
  }
  100% {
    opacity: 1;
    max-height: 96px;
    margin-top: 0;
    padding-top: 0.5rem;
    padding-bottom: 0.5rem;
    transform: translateY(0);
    overflow: visible;
  }
}
.side-panel-ticket-fresh {
  position: relative;
  animation: sidePanelTicketFresh 1100ms cubic-bezier(0.22, 1, 0.36, 1) both;
  will-change: max-height, opacity, transform;
}
/* Hijos del <li> en su propio stacking context para que el flash
   ::before se quede DEBAJO del avatar y los textos sin taparlos. */
.side-panel-ticket-fresh > * {
  position: relative;
  z-index: 1;
}
/* Flash UX de "éxito y acabado": un overlay indigo clarito aparece
   fuerte mientras el item entra (~30–55% de la animación) y se
   difumina a transparente al cerrar. Va en un pseudo-element para
   no pisar el bg base del <li> ni romper su :hover de Tailwind. */
.side-panel-ticket-fresh::before {
  content: "";
  position: absolute;
  inset: 0;
  border-radius: 0.5rem; /* mismo que el .rounded-lg del <li> */
  background-color: rgba(99, 102, 241, 0.28); /* indigo-500 @ 28% — visible */
  pointer-events: none;
  z-index: 0;
  animation: sidePanelTicketFlash 1100ms cubic-bezier(0.22, 1, 0.36, 1) both;
}
@keyframes sidePanelTicketFlash {
  0%   { opacity: 0;    background-color: rgba(99, 102, 241, 0.32); }
  20%  { opacity: 0;    background-color: rgba(99, 102, 241, 0.32); }
  45%  { opacity: 1;    background-color: rgba(99, 102, 241, 0.32); }
  70%  { opacity: 0.55; background-color: rgba(129, 140, 248, 0.22); }
  100% { opacity: 0;    background-color: rgba(165, 180, 252, 0); }
}
.side-panel-ticket-fresh .dp-side-panel-verified-badge {
  animation: sidePanelVerifiedPop 640ms cubic-bezier(0.34, 1.56, 0.64, 1) both;
  animation-delay: 400ms;
}
@keyframes sidePanelVerifiedPop {
  0%   { transform: scale(0.4); opacity: 0; }
  60%  { transform: scale(1.12); opacity: 1; }
  100% { transform: scale(1);    opacity: 1; }
}

@media (prefers-reduced-motion: reduce) {
  .side-panel-activity-fresh,
  .side-panel-activity-item,
  .side-panel-ticket-item,
  .side-panel-ticket-fresh,
  .side-panel-ticket-fresh .dp-side-panel-verified-badge,
  .dp-side-panel-verified-badge,
  .dashboard-grid #dashboard_side_panel { animation: none; transition: none; }
}

/* ── Notificación destacada (deep-link desde campanita) ────────────────────
   Cuando el usuario clica una notificación de tipo `attachment_rejected` la
   campanita lleva a /notifications?highlight=<id>#notification_<id>. El
   stimulus controller notification-highlight añade esta clase 3.5s para
   resaltar visualmente el item al que se acaba de hacer scroll. */
@keyframes notificationHighlight {
  0%   { background-color: rgba(251, 191, 36, 0.25); box-shadow: inset 0 0 0 2px rgba(245, 158, 11, 0.55); }
  60%  { background-color: rgba(251, 191, 36, 0.15); box-shadow: inset 0 0 0 2px rgba(245, 158, 11, 0.35); }
  100% { background-color: transparent;             box-shadow: inset 0 0 0 2px transparent; }
}
.notification-highlighted {
  animation: notificationHighlight 3500ms cubic-bezier(0.22, 1, 0.36, 1) both;
}
@media (prefers-reduced-motion: reduce) {
  .notification-highlighted { animation: none; background-color: rgba(251, 191, 36, 0.18); }
}

/* Cards de resumen dentro del hero — glass embebido */
.dp-hero-stats #summary_cards {
  margin-bottom: 0;
  gap: 16px;
}
.dp-hero-stats .dp-card {
  background: rgba(255, 255, 255, 0.72);
  border: 1px solid rgba(255, 255, 255, 0.85);
  backdrop-filter: blur(14px) saturate(135%);
  -webkit-backdrop-filter: blur(14px) saturate(135%);
  border-radius: 16px;
  padding: 18px;
  box-shadow:
    0 1px 0 rgba(255, 255, 255, 0.5) inset,
    0 6px 18px rgba(79, 70, 229, 0.08);
}

/* Pill de sync activa dentro del hero */
.dp-sync-pill {
  display: inline-flex;
  align-items: center;
  gap: 6px;
  padding: 4px 10px;
  background: rgba(255, 255, 255, 0.6);
  border-radius: 999px;
  backdrop-filter: blur(10px);
  -webkit-backdrop-filter: blur(10px);
  color: #4c1d95;
  font-size: 12px;
  font-weight: 500;
}
.dp-sync-pill .dot {
  width: 6px; height: 6px; border-radius: 50%;
  background: #86efac; box-shadow: 0 0 8px #86efac;
}

/* ── Sensitive field: enmascaramiento visual sin type="password" ────────────── */
.sensitive-masked {
  -webkit-text-security: disc;
  text-security: disc;
}

/* ── Ticket verified: reaparece abajo difuminándose + pulso verde sutil ─────── */
@keyframes ticketVerifiedIn {
  0%   { opacity: 0;   filter: blur(6px); transform: translateY(-3px);
         background-color: rgb(16 185 129 / 0.06); }
  40%  { opacity: 0.7; filter: blur(2px); transform: translateY(0);
         background-color: rgb(16 185 129 / 0.10); }
  100% { opacity: 1;   filter: blur(0);   transform: translateY(0);
         background-color: transparent; }
}

.animate-ticket-verified {
  animation: ticketVerifiedIn 1.6s cubic-bezier(0.16, 1, 0.3, 1) both;
}

/* ── Superficie elevada: separa cajas/tablas del fondo ─────────────────────── */
.surface-elevated {
  border: 1px solid rgb(148 163 184 / 0.24);
  box-shadow:
    0 1px 0 rgb(255 255 255 / 0.78) inset,
    0 10px 24px rgb(15 23 42 / 0.16),
    0 3px 10px rgb(15 23 42 / 0.12);
}

.summary-glass-card {
  /* Glass with low-fill opacity so background stays visible */
  background: linear-gradient(
    145deg,
    rgb(255 255 255 / 0.18) 0%,
    rgb(255 255 255 / 0.1) 42%,
    rgb(255 255 255 / 0.06) 100%
  );
  backdrop-filter: blur(16px) saturate(155%) contrast(112%);
  -webkit-backdrop-filter: blur(16px) saturate(155%) contrast(112%);
  border: 1px solid rgb(255 255 255 / 0.28);
  box-shadow:
    0 1px 0 rgb(255 255 255 / 0.64) inset,
    0 0 0 1px rgb(255 255 255 / 0.12) inset,
    0 14px 28px rgb(15 23 42 / 0.12),
    0 5px 12px rgb(15 23 42 / 0.09),
    0 -6px 14px rgb(255 255 255 / 0.08) inset;
}

/* ── Drawer nav items — estilo sobrio workspace ───────────────────────────── */
.drawer-nav-item {
  display: flex;
  align-items: center;
  gap: 0.875rem;
  margin: 0 0.75rem;
  padding: 0.625rem 0.875rem;
  border-radius: 0.625rem;
  font-size: 0.9375rem;
  font-weight: 500;
  color: rgb(71 85 105);
  transition: background 150ms ease, color 150ms ease, box-shadow 150ms ease;
}

.drawer-nav-item svg {
  flex-shrink: 0;
  width: 1.125rem;
  height: 1.125rem;
  color: rgb(100 116 139);
  transition: color 150ms ease;
}

.drawer-nav-item:hover {
  background: rgba(255, 255, 255, 0.65);
  color: rgb(30 41 59);
}

.drawer-nav-item:hover svg {
  color: rgb(71 85 105);
}

.drawer-nav-item.active {
  background: rgb(255 255 255);
  color: rgb(37 99 235);
  font-weight: 600;
  box-shadow: 0 1px 2px rgba(15, 23, 42, 0.04),
              0 1px 3px rgba(15, 23, 42, 0.06);
}

.drawer-nav-item.active svg {
  color: rgb(37 99 235);
}

.drawer-nav-item.danger:hover {
  background: rgba(254, 242, 242, 0.85);
  color: rgb(185 28 28);
}

.drawer-nav-item.danger:hover svg {
  color: rgb(220 38 38);
}

@media (min-width: 768px) {
  .surface-elevated {
    box-shadow:
      0 1px 0 rgb(255 255 255 / 0.86) inset,
      0 20px 44px rgb(15 23 42 / 0.2),
      0 6px 16px rgb(15 23 42 / 0.15);
  }

  .summary-glass-card {
    backdrop-filter: blur(20px) saturate(165%) contrast(114%);
    -webkit-backdrop-filter: blur(20px) saturate(165%) contrast(114%);
    box-shadow:
      0 1px 0 rgb(255 255 255 / 0.7) inset,
      0 0 0 1px rgb(255 255 255 / 0.14) inset,
      0 20px 40px rgb(15 23 42 / 0.14),
      0 7px 16px rgb(15 23 42 / 0.1),
      0 -8px 18px rgb(255 255 255 / 0.1) inset;
  }
}

/* ── Modal frame: spinner de carga mientras el frame está ocupado ──────────── */
/* Turbo añade el atributo [busy] al turbo-frame mientras carga su contenido. */
turbo-frame#modal[busy] #modal_loading_indicator {
  display: flex !important;
}

/* ── Turbo Frames: shimmer global mientras cargan ──────────────────────────── */
/* Feedback visual uniforme para cualquier turbo-frame lazy/navegado. Evita que
   el usuario tenga la sensación de "clicó pero no pasa nada" durante la carga.
   Se excluye #modal porque tiene su propio indicator (arriba). */
@keyframes turboFrameShimmer {
  0%   { background-position: -180% 0; }
  100% { background-position: 180% 0; }
}

turbo-frame[busy]:not(#modal) {
  position: relative;
  pointer-events: none;
}

turbo-frame[busy]:not(#modal)::after {
  content: "";
  position: absolute;
  inset: 0;
  border-radius: inherit;
  background: linear-gradient(
    100deg,
    rgb(241 245 249 / 0) 35%,
    rgb(226 232 240 / 0.55) 50%,
    rgb(241 245 249 / 0) 65%
  );
  background-size: 200% 100%;
  animation: turboFrameShimmer 1.2s ease-in-out infinite;
  pointer-events: none;
  z-index: 1;
}

turbo-frame[busy]:not(#modal) > * {
  opacity: 0.55;
  transition: opacity 160ms ease;
}

/* ── flag-chip: estados hover/press/selected ─────────────────────────────── */
.flag-chip {
  cursor: pointer;
  user-select: none;
  perspective: 600px;
  transform: rotate(var(--stamp-rot, 0deg));
  transition:
    transform    160ms cubic-bezier(0.34, 1.56, 0.64, 1),
    box-shadow   180ms ease,
    background-color 180ms ease;
}
.flag-chip:hover          { transform: scale(1.4) rotate(var(--stamp-rot, 0deg)); z-index: 10; }
.flag-chip:active         { transform: scale(0.91) rotate(var(--stamp-rot, 0deg)); }
.flag-chip--selected {
  transform: scale(1.3) rotate(var(--stamp-rot, 0deg));
  z-index: 10;
  background-color: rgb(99 102 241 / 0.11);
  box-shadow: 0 0 0 2px rgb(99 102 241 / 0.55), 0 3px 8px -1px rgb(99 102 241 / 0.20);
}
.flag-chip--selected:hover  { transform: scale(1.4) rotate(var(--stamp-rot, 0deg)); }
.flag-chip--selected:active { transform: scale(0.91) rotate(var(--stamp-rot, 0deg)); }

/* ── flag-chip: stamp animation on load ──────────────────────────────────── */
@keyframes stampIn {
  0%   { opacity: 0;   transform: scale(1.35) rotate(var(--stamp-rot, 0deg)); }
  40%  { opacity: 0.7; transform: scale(1.0)  rotate(var(--stamp-rot, 0deg)); }
  65%  { opacity: 1;   transform: scale(0.96) rotate(var(--stamp-rot, 0deg)); }
  82%  { transform: scale(1.02) rotate(var(--stamp-rot, 0deg)); }
  100% { opacity: 1;   transform: scale(1)    rotate(var(--stamp-rot, 0deg)); }
}

.flag-chip--stamping {
  opacity: 0;
  animation: stampIn 800ms cubic-bezier(0.25, 0.46, 0.45, 0.94) both;
}

/* ── flag-chip: 3D flip front/back faces ─────────────────────────────────── */
.flag-chip-inner {
  position: relative;
  display: inline-flex;
  align-items: center;
  width: 100%;
  transition: transform 0.45s cubic-bezier(0.4, 0, 0.2, 1);
  transform-style: preserve-3d;
}

.flag-chip-front,
.flag-chip-back {
  backface-visibility: hidden;
  -webkit-backface-visibility: hidden;
  display: inline-flex;
  align-items: center;
}

.flag-chip-back {
  position: absolute;
  inset: 0;
  transform: rotateY(180deg);
  justify-content: center;
  gap: 0.25rem;
  padding: 0 0.5rem;
}

@media (hover: hover) {
  .flag-chip:hover .flag-chip-inner { transform: rotateY(180deg); }
}

.flag-chip-bar {
  width: 24px; height: 3px;
  background: rgb(226 232 240 / 0.6);
  border-radius: 999px;
  overflow: hidden;
}
.flag-chip-bar-fill {
  height: 100%;
  border-radius: 999px;
}

/* ── country-mosaic-card: trips index flag mosaic glassmorphism ────────────── */
.country-mosaic-card {
  border: 1px solid rgb(255 255 255 / 0.45);
  box-shadow:
    0 1px 0 rgb(255 255 255 / 0.6) inset,
    0 8px 20px rgb(15 23 42 / 0.10),
    0 2px 6px rgb(15 23 42 / 0.08);
  transition: transform 220ms ease, box-shadow 220ms ease;
}
.country-mosaic-card:hover {
  transform: translateY(-3px) scale(1.01);
  box-shadow:
    0 1px 0 rgb(255 255 255 / 0.7) inset,
    0 16px 32px rgb(15 23 42 / 0.16),
    0 4px 10px rgb(15 23 42 / 0.10);
}

/* ── Timeline (movidos desde _country_time_bar.html.erb) ──────────────────── */
.tl-card {
  position: relative;
  background: rgb(255 255 255 / 0.80);
  backdrop-filter: blur(14px) saturate(140%);
  -webkit-backdrop-filter: blur(14px) saturate(140%);
  border: 1px solid rgb(255 255 255 / 0.55);
  border-radius: 20px;
  box-shadow:
    0 1px 0 rgb(255 255 255 / 0.64) inset,
    0 0 0 1px rgb(255 255 255 / 0.12) inset,
    0 14px 28px rgb(15 23 42 / 0.12),
    0 5px 12px rgb(15 23 42 / 0.09),
    0 -6px 14px rgb(255 255 255 / 0.08) inset;
}
@media (min-width: 768px) {
  .tl-card {
    box-shadow:
      0 1px 0 rgb(255 255 255 / 0.7) inset,
      0 0 0 1px rgb(255 255 255 / 0.14) inset,
      0 20px 40px rgb(15 23 42 / 0.14),
      0 7px 16px rgb(15 23 42 / 0.1),
      0 -8px 18px rgb(255 255 255 / 0.1) inset;
  }
}
.tl-track-shell { position: relative; }
.tl-track {
  height: 80px;
  border-radius: 14px;
  background: rgb(255 255 255 / 0.55);
  backdrop-filter: blur(6px);
  -webkit-backdrop-filter: blur(6px);
  border: 1px solid rgb(255 255 255 / 0.45);
  overflow: hidden;
  display: grid;
  grid-template-columns: var(--timeline-columns);
  min-width: 0;
}
.tl-segment {
  position: relative;
  min-width: 0;
  width: calc(100% + 1px);
  margin-right: -1px;
  box-sizing: border-box;
}
.tl-segment--spacer { background: transparent; }
.tl-segment--gap {
  background: repeating-linear-gradient(
    -45deg,
    rgb(254 252 232 / 0.55),
    rgb(254 252 232 / 0.55) 3px,
    rgb(254 249 195 / 0.40) 3px,
    rgb(254 249 195 / 0.40) 6px
  );
  min-width: 18px;
  border-left: 1.5px dashed rgb(217 119 6 / 0.55);
  border-right: 1.5px dashed rgb(217 119 6 / 0.55);
}
.tl-segment--trip {
  background: linear-gradient(to right, var(--seg-color) 0%, var(--seg-fade) 100%);
}
.tl-segment--trip::before {
  content: "";
  position: absolute;
  left: 0; top: 0; bottom: 0;
  width: 4px;
  background: var(--seg-vivid);
  pointer-events: none;
  z-index: 2;
}
.tl-segment--trip::after {
  content: "";
  position: absolute;
  inset: 0;
  background: transparent;
  transition: background 140ms ease;
  pointer-events: none;
}
.tl-segment--trip:hover::after { background: rgb(0 0 0 / 0.07); }
.tl-month-dividers {
  position: absolute;
  inset: 1px;
  border-radius: 13px;
  overflow: hidden;
  pointer-events: none;
}

/* ── Timeline — clases adicionales (de _country_time_bar.html.erb) ─────────── */
.tl-month-tick {
  position: absolute;
  top: 0; bottom: 0;
  width: 1px;
  background: rgb(255 255 255 / 0.55);
}
.tl-flag-chip {
  background: rgb(255 255 255 / 0.82);
  border: 1px solid rgb(255 255 255 / 0.60);
  border-radius: 6px;
}
.tl-axis-row {
  position: relative;
  width: 100%;
  height: 1.25rem;
  margin-top: 0.5rem;
}
.tl-rule {
  height: 1px;
  background: rgb(226 232 240);
  margin: 1.125rem 0 0.875rem;
}
.tl-legend {
  display: flex;
  flex-wrap: wrap;
  gap: 0.375rem 0.5rem;
}
.tl-legend-chip {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.28rem 0.7rem 0.28rem 0.45rem;
  border-radius: 999px;
  border: 1px solid rgb(226 232 240);
  background: rgb(248 250 252);
}
.tl-legend-swatch {
  width: 9px; height: 9px;
  border-radius: 3px;
  flex-shrink: 0;
}
/* Timeline highlight animation */
.tl-segment--trip {
  cursor: pointer;
  transition:
    opacity 420ms cubic-bezier(0.4, 0, 0.2, 1),
    filter  420ms cubic-bezier(0.4, 0, 0.2, 1);
}
.tl-segment--highlighted {
  filter: saturate(2.2) brightness(1.13) drop-shadow(0 0 10px rgba(0,0,0,0.20));
  z-index: 5;
  transition:
    filter  340ms cubic-bezier(0, 0, 0.2, 1),
    opacity 340ms cubic-bezier(0, 0, 0.2, 1);
}
.tl-segment--dimmed {
  opacity: 0.13;
  filter: saturate(0.2) grayscale(0.35) brightness(0.88);
  transition:
    opacity 480ms cubic-bezier(0.4, 0, 0.8, 1),
    filter  480ms cubic-bezier(0.4, 0, 0.8, 1);
}
.tl-segment--gap-dimmed {
  opacity: 0.28;
  transition: opacity 480ms cubic-bezier(0.4, 0, 0.8, 1);
}
.tl-legend-chip[data-timeline-highlight-target~="legendChip"] {
  cursor: pointer;
  user-select: none;
  transition: opacity 380ms cubic-bezier(0.4, 0, 0.2, 1), box-shadow 380ms ease;
}
.tl-legend-chip--active {
  box-shadow: 0 0 0 2px rgb(100 116 139 / 0.45);
  transition: opacity 300ms cubic-bezier(0, 0, 0.2, 1), box-shadow 300ms ease;
}
.tl-legend-chip--inactive {
  opacity: 0.20;
  transition: opacity 480ms cubic-bezier(0.4, 0, 0.8, 1), box-shadow 480ms ease;
}

/* Timeline flag watermark */
.tl-flag-watermark {
  position: absolute;
  inset: 0;
  border-radius: 18px;
  pointer-events: none;
  background-image: var(--tl-flag-url, none);
  background-size: 52%;
  background-position: 92% center;
  background-repeat: no-repeat;
  opacity: 0;
  transition: opacity 350ms ease;
  z-index: 2;
}
.tl-flag-watermark--visible {
  opacity: 0.085;
}

/* Timeline collapsible body */
.tl-body {
  overflow: hidden;
  max-height: 400px;
  opacity: 1;
  transition:
    max-height 380ms cubic-bezier(0.4, 0, 0.2, 1),
    opacity    280ms ease;
}
.tl-body--collapsed {
  max-height: 0;
  opacity: 0;
  transition:
    max-height 320ms cubic-bezier(0.4, 0, 0.6, 1),
    opacity    200ms ease;
}

/* ── Fiscal Report Ready — banner celebration animations ─────────────────── */

/* Banner entrada: scale + ligero bounce */
@keyframes fiscalCelebrationIn {
  0%   { opacity: 0; transform: scale(0.94) translateY(-6px); }
  55%  { opacity: 1; transform: scale(1.015) translateY(0); }
  80%  { transform: scale(0.995); }
  100% { opacity: 1; transform: scale(1); }
}
.animate-fiscal-celebration {
  animation: fiscalCelebrationIn 0.7s cubic-bezier(0.22, 1, 0.36, 1) both;
}

/* Check icon pop — escala al entrar con overshoot sutil */
@keyframes fiscalCheckPop {
  0%   { transform: scale(0.3) rotate(-15deg); }
  60%  { transform: scale(1.12) rotate(4deg); }
  100% { transform: scale(1) rotate(0); }
}
.animate-fiscal-check-pop {
  animation: fiscalCheckPop 0.65s cubic-bezier(0.34, 1.56, 0.64, 1) 0.15s both;
}

/* Ring burst — anillos que se expanden y desvanecen */
@keyframes fiscalRingBurst {
  0%   { transform: scale(1);   opacity: 0.85; }
  100% { transform: scale(2.4); opacity: 0;    }
}
.animate-fiscal-ring-burst {
  animation: fiscalRingBurst 1.1s ease-out 0.2s both;
}

/* Confetti — partículas que caen con rotación */
@keyframes fiscalConfetti {
  0%   { opacity: 0; transform: translate(0, -12px) scale(0.8); }
  15%  { opacity: 1; }
  100% { opacity: 0; transform: translate(var(--confetti-dx, 8px), 40px) scale(1.1); }
}
.animate-fiscal-confetti {
  animation: fiscalConfetti 1.6s cubic-bezier(0.33, 0.1, 0.67, 1) both;
}

/* ── Analysis Banner: 20 segmentos crecientes + pulso blanco interno ─────── */
.analysis-progress-track {
  height: 6px;
  background: rgba(255, 255, 255, 0.55);
  backdrop-filter: blur(6px);
  -webkit-backdrop-filter: blur(6px);
  border: 1px solid rgba(255, 255, 255, 0.45);
  border-radius: 999px;
  overflow: hidden;
  position: relative;
  padding: 0 2px;
}

.analysis-progress-segments {
  display: flex;
  gap: 2px;
  height: 100%;
  width: 100%;
}

.analysis-progress-segment {
  flex: 1;
  background: rgba(99, 102, 241, 0.2);
  border-radius: 2px;
  position: relative;
  overflow: hidden;
}

.analysis-progress-segment.is-active {
  background: #6366f1;
  animation:
    analysisSegmentGrow 0.5s cubic-bezier(0.34, 1.56, 0.64, 1) both,
    analysisSegmentPulseY 1.5s ease-in-out infinite;
  animation-delay:
    var(--grow-delay, 0s),
    calc(var(--grow-delay, 0s) + var(--pulse-delay, 0s) + 0.5s);
}

@keyframes analysisSegmentGrow {
  0%   { transform: scaleY(0.3); opacity: 0; }
  100% { transform: scaleY(1);   opacity: 1; }
}

@keyframes analysisSegmentPulseY {
  0%, 100% { transform: scaleY(1);   opacity: 1;   }
  50%      { transform: scaleY(0.4); opacity: 0.6; }
}

/* ── Batch Celebration — la clase .is-celebrating se aplica al root de    ── */
/* #ticket_table. Los descendientes (batch-celebration, batch-celebration-   */
/* check, batch-celebration-text) reaccionan vía selector descendente.      */
.is-celebrating .batch-celebration {
  animation: batchCelebrationEntrance 0.9s cubic-bezier(0.22, 1, 0.36, 1) both;
  transform-origin: left center;
}

@keyframes batchCelebrationEntrance {
  0%   { opacity: 0; transform: scale(0.96); background-color: rgb(167 243 208 / 0.55); }
  40%  { opacity: 1; background-color: rgb(167 243 208 / 0.45); }
  100% { transform: scale(1); background-color: rgb(224 231 255 / 0.5); }
}

/* Check — pop con overshoot, estilo fiscalCheckPop */
.is-celebrating .batch-celebration-check {
  animation: batchCelebrationCheckPop 0.7s cubic-bezier(0.34, 1.56, 0.64, 1) 0.3s both;
}

@keyframes batchCelebrationCheckPop {
  0%   { transform: scale(0.35) rotate(-18deg); }
  60%  { transform: scale(1.18) rotate(5deg);   }
  100% { transform: scale(1)    rotate(0);      }
}

/* Ring burst — anillo que crece y se desvanece */
.is-celebrating .batch-celebration-check::before {
  content: '';
  position: absolute;
  inset: -3px;
  border: 2px solid rgb(99 102 241 / 0.55);
  border-radius: 50%;
  pointer-events: none;
  animation: batchCelebrationRingBurst 1s ease-out 0.4s both;
}

@keyframes batchCelebrationRingBurst {
  0%   { transform: scale(1);   opacity: 0.85; }
  100% { transform: scale(2.6); opacity: 0;    }
}

/* Text slide — entra desde abajo con blur */
.is-celebrating .batch-celebration-text {
  animation: batchCelebrationTextSlide 0.6s cubic-bezier(0.22, 1, 0.36, 1) 0.45s both;
}

/* Confetti particles — ocultas por defecto, animan al celebrar.
   Animación propia (batchConfetti) con amplitud vertical 90px y halo glow para
   que destaquen sobre el gradiente violet/white del panel; reusar fiscalConfetti
   con 40px de caída dejaba los puntos casi imperceptibles. */
.batch-confetti-particle {
  opacity: 0;
  box-shadow: 0 0 0 1px rgba(255, 255, 255, 0.55), 0 0 6px currentColor;
  will-change: transform, opacity;
}

@keyframes batchConfetti {
  0%   { opacity: 0; transform: translate(0, -16px) scale(0.6); }
  12%  { opacity: 1; transform: translate(calc(var(--confetti-dx, 14px) * 0.2), 8px) scale(1); }
  60%  { opacity: 1; transform: translate(calc(var(--confetti-dx, 14px) * 0.7), 50px) scale(1.05); }
  100% { opacity: 0; transform: translate(var(--confetti-dx, 14px), 90px) scale(1.1); }
}

.is-celebrating .batch-confetti-particle {
  animation: batchConfetti 2.4s cubic-bezier(0.33, 0.1, 0.67, 1) infinite;
}

@keyframes batchCelebrationTextSlide {
  0%   { opacity: 0; transform: translateY(6px); filter: blur(3px); }
  100% { opacity: 1; transform: translateY(0);   filter: blur(0);   }
}

/* Accesibilidad — respetar prefers-reduced-motion */
@media (prefers-reduced-motion: reduce) {
  .is-celebrating .batch-celebration,
  .is-celebrating .batch-celebration-check,
  .is-celebrating .batch-celebration-check::before,
  .is-celebrating .batch-celebration-text,
  .is-celebrating .batch-confetti-particle {
    animation: none !important;
  }
}

/* ── Identity success — celebración tras confirmar nombre oficial     ── */
/* Se renderiza dentro de #flash_messages. El controller identity_success */
/* aplica .is-entering al connect y .is-leaving tras ~1.9s.               */
.identity-success-card {
  opacity: 0;
  transform: scale(0.6);
}

.identity-success-card.is-entering {
  animation: identitySuccessEnter 0.35s cubic-bezier(0.34, 1.56, 0.64, 1) both;
}

@keyframes identitySuccessEnter {
  0%   { opacity: 0; transform: scale(0.6);  }
  60%  { opacity: 1; transform: scale(1.04); }
  100% { opacity: 1; transform: scale(1);    }
}

.identity-success-card.is-leaving {
  animation: identitySuccessExit 0.4s ease-in both;
}

@keyframes identitySuccessExit {
  0%   { opacity: 1; transform: scale(1);    }
  100% { opacity: 0; transform: scale(1.05); }
}

/* Check grande — pop con overshoot, arranca un pelín después que la card */
.identity-success-card.is-entering .identity-success-check {
  animation: identitySuccessCheckPop 0.45s cubic-bezier(0.34, 1.56, 0.64, 1) 0.2s both;
}

@keyframes identitySuccessCheckPop {
  0%   { transform: scale(0.4) rotate(-10deg); }
  60%  { transform: scale(1.15) rotate(4deg);  }
  100% { transform: scale(1) rotate(0);         }
}

/* Ring burst verde alrededor del check */
.identity-success-card.is-entering .identity-success-check::before {
  content: '';
  position: absolute;
  inset: -4px;
  border: 2px solid rgb(16 185 129 / 0.55);
  border-radius: 50%;
  pointer-events: none;
  animation: identitySuccessRingBurst 0.9s ease-out 0.3s both;
}

@keyframes identitySuccessRingBurst {
  0%   { transform: scale(1);   opacity: 0.85; }
  100% { transform: scale(2.3); opacity: 0;    }
}

/* Texto — slide sutil desde abajo con blur */
.identity-success-card.is-entering .identity-success-text {
  animation: identitySuccessTextSlide 0.45s cubic-bezier(0.22, 1, 0.36, 1) 0.35s both;
}

@keyframes identitySuccessTextSlide {
  0%   { opacity: 0; transform: translateY(6px); filter: blur(2px); }
  100% { opacity: 1; transform: translateY(0);   filter: blur(0);   }
}

/* Micro-pulso verde en #ticket_table — señal de "aquí pasó algo" */
.identity-row-pulse {
  animation: identityRowPulse 0.7s ease-out;
}

@keyframes identityRowPulse {
  0%   { box-shadow: 0 0 0 0 rgb(34 197 94 / 0.35); }
  100% { box-shadow: 0 0 0 14px rgb(34 197 94 / 0); }
}

/* Accesibilidad — prefers-reduced-motion: colapsar a fade simple */
@media (prefers-reduced-motion: reduce) {
  .identity-success-card.is-entering {
    animation: identitySuccessFadeIn 0.15s ease-out both;
  }
  .identity-success-card.is-leaving {
    animation: identitySuccessFadeOut 0.15s ease-out both;
  }
  .identity-success-card.is-entering .identity-success-check,
  .identity-success-card.is-entering .identity-success-check::before,
  .identity-success-card.is-entering .identity-success-text,
  .identity-row-pulse {
    animation: none !important;
  }

  @keyframes identitySuccessFadeIn {
    0%   { opacity: 0; transform: scale(1); }
    100% { opacity: 1; transform: scale(1); }
  }
  @keyframes identitySuccessFadeOut {
    0%   { opacity: 1; transform: scale(1); }
    100% { opacity: 0; transform: scale(1); }
  }
}

/* ── Name banner — colapso al guardar el nombre oficial desde el banner ── */
/* El Stimulus name_banner_collapse aplica .is-collapsing + max-height:0     */
/* antes de que Turbo ejecute el remove real. Así el contenido debajo sube   */
/* con una transición suave en vez de un corte instantáneo. Transición más  */
/* lenta (620ms) para una sensación menos apresurada.                        */
.name-banner-shell {
  transition:
    max-height 0.62s cubic-bezier(0.4, 0, 0.2, 1),
    opacity    0.45s ease-out 0.1s,
    transform  0.62s cubic-bezier(0.4, 0, 0.2, 1);
  transform-origin: top center;
  will-change: max-height, opacity, transform;
}

.name-banner-shell.is-collapsing {
  opacity: 0;
  transform: scaleY(0.92);
}

/* ── Name success burst — ticket verde OK grande, centrado, "desde el   ── */
/* fondo". Patrón Material Emphasized Decelerate: scale desde 0.4 con blur  */
/* y overshoot ligero → estado pleno. Hold de 1s (orquestado en JS) y        */
/* salida rápida ease-in. Los selectores parten de `#name_success_burst`    */
/* porque la clase .is-entering la aplica el Stimulus al outer overlay —   */
/* la card interna tiene .name-success-burst pero no recibe la flag.       */
#name_success_burst .name-success-burst {
  opacity: 0;
  transform: scale(0.4);
  filter: blur(8px);
}

#name_success_burst.is-entering .name-success-burst {
  animation: nameSuccessBurstEnter 0.6s cubic-bezier(0.34, 1.56, 0.64, 1) both;
}

@keyframes nameSuccessBurstEnter {
  0%   { opacity: 0; transform: scale(0.4);  filter: blur(8px); }
  55%  { opacity: 1; transform: scale(1.08); filter: blur(0);   }
  100% { opacity: 1; transform: scale(1);    filter: blur(0);   }
}

#name_success_burst.is-leaving .name-success-burst {
  animation: nameSuccessBurstLeave 0.4s cubic-bezier(0.4, 0, 1, 1) both;
}

@keyframes nameSuccessBurstLeave {
  0%   { opacity: 1; transform: scale(1);    }
  100% { opacity: 0; transform: scale(1.12); }
}

/* Check grande — pop con overshoot encima del spring del contenedor */
#name_success_burst.is-entering .name-success-burst-check {
  animation: nameSuccessBurstCheckPop 0.5s cubic-bezier(0.34, 1.56, 0.64, 1) 0.2s both;
}

@keyframes nameSuccessBurstCheckPop {
  0%   { transform: scale(0.3) rotate(-14deg); }
  60%  { transform: scale(1.15) rotate(6deg);  }
  100% { transform: scale(1)    rotate(0);     }
}

/* Ring burst verde alrededor del check */
#name_success_burst.is-entering .name-success-burst-check::before {
  content: '';
  position: absolute;
  inset: -4px;
  border: 2px solid rgb(16 185 129 / 0.55);
  border-radius: 50%;
  pointer-events: none;
  animation: nameSuccessBurstRingBurst 1s ease-out 0.35s both;
}

@keyframes nameSuccessBurstRingBurst {
  0%   { transform: scale(1);   opacity: 0.85; }
  100% { transform: scale(2.4); opacity: 0;    }
}

/* Texto — slide sutil desde abajo tras el check */
#name_success_burst.is-entering .name-success-burst-label {
  animation: nameSuccessBurstLabel 0.45s cubic-bezier(0.22, 1, 0.36, 1) 0.4s both;
}

@keyframes nameSuccessBurstLabel {
  0%   { opacity: 0; transform: translateY(8px); filter: blur(2px); }
  100% { opacity: 1; transform: translateY(0);   filter: blur(0);   }
}

/* ── Greeting welcome — entrada animada tras guardar el nombre oficial ── */
/* El greeting permanece invisible hasta que el Stimulus añade              */
/* .is-welcoming (tras el hold del success burst). En cargas normales del   */
/* dashboard la clase `.greeting-welcome` no aparece → sin efectos.         */
.greeting-welcome {
  opacity: 0;
}
.greeting-welcome.is-welcoming {
  opacity: 1;
  transition: opacity 0.4s ease-out;
}

.greeting-welcome.is-welcoming .greeting-welcome-text {
  animation: greetingWelcomeTextSlide 0.5s cubic-bezier(0.22, 1, 0.36, 1) both;
}

@keyframes greetingWelcomeTextSlide {
  0%   { opacity: 0; transform: translateY(10px); filter: blur(4px); }
  100% { opacity: 1; transform: translateY(0);    filter: blur(0);   }
}

.greeting-welcome.is-welcoming .greeting-welcome-check {
  animation: greetingWelcomeCheckPop 0.55s cubic-bezier(0.34, 1.56, 0.64, 1) 0.25s both;
}

@keyframes greetingWelcomeCheckPop {
  0%   { opacity: 0; transform: scale(0.4) rotate(-12deg); }
  60%  { opacity: 1; transform: scale(1.18) rotate(6deg);  }
  100% { opacity: 1; transform: scale(1)    rotate(0);     }
}

/* Ring burst verde alrededor del check del greeting */
.greeting-welcome.is-welcoming .greeting-welcome-check::before {
  content: '';
  position: absolute;
  inset: -3px;
  border: 2px solid rgb(16 185 129 / 0.55);
  border-radius: 50%;
  pointer-events: none;
  animation: greetingWelcomeRingBurst 0.8s ease-out 0.35s both;
}

@keyframes greetingWelcomeRingBurst {
  0%   { transform: scale(1);   opacity: 0.8; }
  100% { transform: scale(2.1); opacity: 0;   }
}

/* Estado final limpio tras que el controller quita .is-welcoming */
.greeting-welcome-check {
  position: relative;
}

/* ── Name saved toast — pill verde con entrada slide desde la derecha. ── */
/* Delay 1.4s para no competir con el name_success_burst dominante y caer  */
/* cuando éste está saliendo (el foco visual se traslada del centro al     */
/* toast periférico con naturalidad).                                       */
.name-saved-toast {
  opacity: 0;
  animation: nameSavedToastEnter 0.4s cubic-bezier(0.22, 1, 0.36, 1) 1.4s both;
  transform-origin: right center;
}

@keyframes nameSavedToastEnter {
  0%   { opacity: 0; transform: translateX(24px) scale(0.95); }
  100% { opacity: 1; transform: translateX(0)    scale(1);    }
}

.name-saved-toast-check {
  animation: nameSavedToastCheckPop 0.45s cubic-bezier(0.34, 1.56, 0.64, 1) 0.15s both;
}

@keyframes nameSavedToastCheckPop {
  0%   { transform: scale(0.5) rotate(-8deg); }
  60%  { transform: scale(1.15); }
  100% { transform: scale(1)    rotate(0); }
}

/* Accesibilidad — prefers-reduced-motion para los tres bloques anteriores */
@media (prefers-reduced-motion: reduce) {
  .name-banner-shell {
    transition: opacity 0.15s ease-out;
  }
  .name-banner-shell.is-collapsing {
    max-height: 0 !important;
    transform: none;
  }
  .greeting-welcome.is-welcoming .greeting-welcome-text,
  .greeting-welcome.is-welcoming .greeting-welcome-check,
  .greeting-welcome.is-welcoming .greeting-welcome-check::before {
    animation: none !important;
  }
  .name-saved-toast,
  .name-saved-toast-check {
    animation: nameSavedToastFadeIn 0.15s ease-out both;
  }
  @keyframes nameSavedToastFadeIn {
    0%   { opacity: 0; }
    100% { opacity: 1; }
  }

  #name_success_burst.is-entering .name-success-burst {
    animation: nameSuccessBurstFadeIn 0.2s ease-out both;
  }
  #name_success_burst.is-leaving .name-success-burst {
    animation: nameSuccessBurstFadeOut 0.2s ease-out both;
  }
  #name_success_burst.is-entering .name-success-burst-check,
  #name_success_burst.is-entering .name-success-burst-check::before,
  #name_success_burst.is-entering .name-success-burst-label {
    animation: none !important;
  }
  @keyframes nameSuccessBurstFadeIn {
    0%   { opacity: 0; transform: scale(1); filter: none; }
    100% { opacity: 1; transform: scale(1); filter: none; }
  }
  @keyframes nameSuccessBurstFadeOut {
    0%   { opacity: 1; }
    100% { opacity: 0; }
  }
}

/* ── Year Start Marker — toasts y microtransiciones ──────────────────────── */
@keyframes year-start-marker-fade-in {
  0%   { opacity: 0; transform: translateY(8px); }
  100% { opacity: 1; transform: translateY(0); }
}
.animate-fade-in {
  animation: year-start-marker-fade-in 240ms ease-out;
}
