/*
 * 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.
 */

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

/* ── Ticket verified: reaparece abajo difuminándose + pulso verde ───────────── */
@keyframes ticketVerifiedIn {
  0%   { opacity: 0;    filter: blur(7px);  transform: translateY(-4px); background-color: rgb(209 250 229 / 0.1); }
  30%  { opacity: 0.6;  filter: blur(3px);  transform: translateY(0);    background-color: rgb(52 211 153 / 0.45); }
  60%  { opacity: 1;    filter: blur(0px);                               background-color: rgb(16 185 129 / 0.35); }
  100% {                filter: blur(0px);                               background-color: rgb(209 250 229 / 0.3); }
}

.animate-ticket-verified {
  animation: ticketVerifiedIn 2.4s 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 (reusa fiscalConfetti) */
.batch-confetti-particle {
  opacity: 0;
}

.is-celebrating .batch-confetti-particle {
  animation: fiscalConfetti 2.2s 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; }
  }
}
