/* =========================================================================
   Scan page — live camera overlay. Полноэкранный stage с живым превью
   камеры, SVG-рамкой (80–85% экрана, пропорции A4) и зонами затемнения.
   Цветовая индикация (зелёный/красный) управляется data-hint на .cam-stage.
   ========================================================================= */

.scan-body {
  margin: 0;
  background: #0b1020;
  min-height: 100vh;
  min-height: 100dvh;
}

/* У .magic-stage и .cam-stage стоит display: flex — он перебивает
   HTML-атрибут hidden. Чиним явно. */
[hidden] { display: none !important; }

.scan-card { max-width: 520px; margin: 2rem auto; }

/* --- Шаг 1 --- */
#go-camera { width: 100%; font-size: 1.05rem; padding: 0.9rem 1rem; }

button.secondary {
  background: transparent;
  color: #cfd9f2;
  border: 1px solid #2a3358;
}

/* --- Шаг 2: camera stage --- */
.cam-stage {
  /* Высоту берём по dvh, не inset:0. На Android Chrome fixed+inset:0
     растягивается на БОЛЬШОЙ viewport (URL-бар скрыт); когда бар
     появляется (часто после поворота) — перекрывает низ stage, и шаттер
     (bottom:24px) уезжает под бар, фото не нажать. env(safe-area-inset-
     bottom) на Android = 0, не помогает. dvh = текущий видимый viewport:
     низ stage = низ экрана над баром, шаттер всегда дотягиваемый. iOS
     Safari dvh уважает — единое поведение. */
  position: fixed;
  top: 0; left: 0; right: 0;
  height: 100vh;   /* fallback для старых браузеров без dvh */
  height: 100dvh;
  background: #000;
  z-index: 50;
  overflow: hidden;

  /* Используем CSS-переменные: меняя --frame-shade / --frame-stroke
     через JS, не трогаем DOM SVG-узлов. */
  --frame-shade: rgba(0, 0, 0, 0.55);
  --frame-stroke: rgba(210, 210, 210, 0.85);
  --frame-corner: rgba(210, 210, 210, 0.95);
}

.cam-stage[data-hint="ok"] {
  --frame-shade: rgba(10, 60, 30, 0.55);
  --frame-stroke: rgba(34, 197, 94, 1);
  --frame-corner: rgba(34, 197, 94, 1);
}
.cam-stage[data-hint="partial"],
.cam-stage[data-hint="no_frame"],
.cam-stage[data-hint="move_closer"],
.cam-stage[data-hint="move_farther"],
.cam-stage[data-hint="too_small"],
.cam-stage[data-hint="low_light"] {
  --frame-shade: rgba(80, 15, 20, 0.55);
  --frame-stroke: rgba(239, 68, 68, 0.95);
  --frame-corner: rgba(239, 68, 68, 0.95);
}

#cam-video {
  position: absolute; inset: 0;
  width: 100%; height: 100%;
  object-fit: cover;
  pointer-events: none;
}

/* .cam-frame всегда портретный A4 (210/297) — ориентация устройства
   намеренно игнорируется. Поворот телефона во время съёмки раньше менял
   aspect-ratio через @media landscape: layout прыгал, marker-overlay
   улетал. Камера-UI orientation-invariant: рамка не флипается, capture
   (video.videoWidth/Height = сенсор) от CSS не зависит. Размер — 85%
   меньшей стороны viewport с капом. Затемнение — гигантский box-shadow. */
.cam-frame {
  position: absolute;
  top: 50%; left: 50%;
  transform: translate(-50%, -50%);
  aspect-ratio: 210 / 297;
  width: min(85vw, 60vh);
  max-height: 85vh;
  border: 2px solid var(--frame-stroke);
  border-radius: 10px;
  /* Гигантский внешний shadow = затемнение вокруг рамки.
     Цвет управляется --frame-shade. */
  box-shadow: 0 0 0 9999px var(--frame-shade);
  transition: border-color 0.2s, box-shadow 0.2s;
  pointer-events: none;
}


/* Угловые маркеры — L-образные линии в каждом углу рамки.
   Это только декоративные L-метки; статусные точки рендерятся через
   .cam-marker (positioned overlay) и подсвечивают сами коды. */
.cam-corner {
  position: absolute;
  width: 28px; height: 28px;
  border: 0 solid var(--frame-corner);
  transition: border-color 0.2s;
}
.cam-corner.tl { top: -3px;    left: -3px;    border-top-width: 4px;    border-left-width: 4px;  border-top-left-radius: 8px; }
.cam-corner.tr { top: -3px;    right: -3px;   border-top-width: 4px;    border-right-width: 4px; border-top-right-radius: 8px; }
.cam-corner.br { bottom: -3px; right: -3px;   border-bottom-width: 4px; border-right-width: 4px; border-bottom-right-radius: 8px; }
.cam-corner.bl { bottom: -3px; left: -3px;    border-bottom-width: 4px; border-left-width: 4px;  border-bottom-left-radius: 8px; }

/* --- Live marker overlay поверх <video> ---
   Точки в координатах video после учёта object-fit: cover crop. JS считает
   позицию и пишет в transform translate(x, y) у каждого .cam-marker.
   Зелёные = код найден на этой позиции, красные = код отсутствует
   (отображается в позиции угла рамки как индикатор «здесь должен быть код»). */
.cam-markers {
  position: absolute; inset: 0;
  pointer-events: none;
  z-index: 5;
}
.cam-marker {
  position: absolute;
  top: 0; left: 0;
  width: 22px; height: 22px;
  margin-left: -11px; margin-top: -11px;
  border-radius: 50%;
  background: rgba(34, 197, 94, 0.9);
  border: 2px solid #fff;
  box-shadow: 0 0 12px rgba(34, 197, 94, 0.8);
  transition: transform 0.12s linear, background 0.2s, box-shadow 0.2s;
}
.cam-marker.missing {
  background: rgba(239, 68, 68, 0.9);
  box-shadow: 0 0 12px rgba(239, 68, 68, 0.8);
  animation: cam-marker-pulse 0.9s ease-in-out infinite;
}
.cam-marker[hidden] { display: none; }

@keyframes cam-marker-pulse {
  0%, 100% { opacity: 1; }
  50%      { opacity: 0.45; }
}

/* --- Превью персонажа в углу --- */
.cam-creature {
  position: absolute;
  top: calc(env(safe-area-inset-top, 0px) + 70px);
  left: 14px;
  display: flex; align-items: center; gap: 0.5rem;
  background: rgba(0, 0, 0, 0.55);
  padding: 0.35rem 0.7rem 0.35rem 0.4rem;
  border-radius: 999px;
  backdrop-filter: blur(6px);
  pointer-events: none;
  max-width: calc(100vw - 28px);
  animation: cam-creature-in 0.25s ease-out;
}
.cam-creature[hidden] { display: none; }
.cam-creature img {
  width: 44px; height: 44px;
  object-fit: contain;
  border-radius: 50%;
  background: rgba(255, 255, 255, 0.08);
}
.cam-creature-name {
  color: white;
  font-size: 0.85rem;
  font-weight: 600;
  text-shadow: 0 1px 2px rgba(0, 0, 0, 0.5);
  white-space: nowrap;
}
@keyframes cam-creature-in {
  from { opacity: 0; transform: translateX(-12px); }
  to   { opacity: 1; transform: translateX(0); }
}

/* Скрываем дублирующий элемент .cam-shade — затемнение создаётся через
   box-shadow у .cam-frame. Оставляем узел в DOM для возможных будущих
   эффектов. */
.cam-shade { display: none; }

.cam-hint {
  position: absolute;
  left: 50%;
  top: calc(env(safe-area-inset-top, 0px) + 18px);
  transform: translateX(-50%);
  color: white;
  font-size: 1rem;
  font-weight: 600;
  padding: 0.55rem 0.9rem;
  border-radius: 999px;
  background: rgba(0, 0, 0, 0.55);
  backdrop-filter: blur(6px);
  text-shadow: 0 1px 2px rgba(0, 0, 0, 0.4);
  max-width: 90vw; text-align: center;
  pointer-events: none;
}

.cam-close {
  position: absolute;
  top: calc(env(safe-area-inset-top, 0px) + 14px);
  right: 14px;
  width: 44px; height: 44px;
  border-radius: 50%;
  border: none;
  background: rgba(0, 0, 0, 0.55);
  color: white; font-size: 24px; line-height: 1;
  cursor: pointer;
  pointer-events: auto;
  display: flex; align-items: center; justify-content: center;
}

.cam-shutter {
  position: absolute;
  left: 50%;
  bottom: calc(env(safe-area-inset-bottom, 0px) + 24px);
  transform: translateX(-50%);
  width: 76px; height: 76px;
  border-radius: 50%;
  border: 4px solid white;
  background: transparent;
  cursor: pointer;
  pointer-events: auto;
  padding: 0;
  display: flex; align-items: center; justify-content: center;
  transition: border-color 0.2s, transform 0.1s;
}
.cam-shutter span {
  display: block;
  width: 60px; height: 60px;
  border-radius: 50%;
  background: white;
  transition: background 0.15s, transform 0.1s;
}
.cam-shutter:not([disabled]):active span {
  background: #ddd; transform: scale(0.9);
}
.cam-shutter[disabled] {
  border-color: rgba(255, 255, 255, 0.45);
  cursor: not-allowed;
}
.cam-shutter[disabled] span { background: rgba(255, 255, 255, 0.45); }

.cam-stage[data-hint="ok"] .cam-shutter {
  border-color: rgb(34, 197, 94);
  box-shadow: 0 0 24px rgba(34, 197, 94, 0.4);
}

/* Auto-shutter countdown — visible feedback во время AUTO_SHUTTER_DELAY_MS.
   Кнопка пульсирует, hint меняет фон на зелёный, чтобы было ясно: сейчас
   снимем сами. Длительность анимации синхронизирована с AUTO_SHUTTER_DELAY_MS
   во flow.js (400ms). */
.cam-shutter.auto-counting {
  animation: cam-shutter-countdown 0.4s linear forwards;
}
.cam-shutter.auto-counting span {
  background: rgb(34, 197, 94);
}
@keyframes cam-shutter-countdown {
  0%   { transform: translateX(-50%) scale(1);    box-shadow: 0 0 24px rgba(34, 197, 94, 0.4); }
  50%  { transform: translateX(-50%) scale(1.08); box-shadow: 0 0 36px rgba(34, 197, 94, 0.85); }
  100% { transform: translateX(-50%) scale(1);    box-shadow: 0 0 48px rgba(34, 197, 94, 1); }
}
.cam-hint.auto-counting {
  background: rgba(34, 197, 94, 0.85);
}

/* --- Шаг 3 --- */
.preview-wrap {
  position: relative;
}
.shot {
  width: 100%;
  border-radius: 10px;
  background: #0a0f22;
  display: block;
  object-fit: contain;
  max-height: 60vh;
}
.preview-loader {
  position: absolute;
  inset: 0;
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  gap: 1rem;
  border-radius: 10px;
  background: rgba(10, 15, 34, 0.7);
  backdrop-filter: blur(4px);
  color: #fff;
  pointer-events: none;
}
.preview-spinner {
  width: 56px;
  height: 56px;
  border-radius: 50%;
  border: 4px solid rgba(255, 255, 255, 0.18);
  border-top-color: #7ee0a1;
  animation: preview-spin 0.9s linear infinite;
}
.preview-loader-text {
  font-size: 1rem;
  font-weight: 600;
  text-shadow: 0 0 12px rgba(120, 170, 255, 0.6);
}
@keyframes preview-spin {
  to { transform: rotate(360deg); }
}

/* --- Шаг 4: magic loader --- */
.magic-stage {
  position: fixed; inset: 0;
  z-index: 60;
  background: radial-gradient(circle at center, #1c2754 0%, #050a1a 75%);
  display: flex; align-items: center; justify-content: center;
  flex-direction: column;
  gap: 1.2rem;
  padding: 2rem;
  text-align: center;
}
.magic-core {
  position: relative;
  width: min(80vw, 320px);
  aspect-ratio: 1 / 1;
  display: flex; align-items: center; justify-content: center;
}
#magic-canvas {
  position: absolute; inset: 0;
  width: 100%; height: 100%;
}
.magic-text {
  position: relative;
  color: #fff;
  font-size: 1.15rem;
  font-weight: 600;
  text-shadow: 0 0 12px rgba(120, 170, 255, 0.6);
  z-index: 2;
  padding: 0 1rem;
}
.magic-result { color: #cfd9f2; min-height: 1.5rem; }
.magic-result .error { color: #ff7a90; }
.magic-result .ok { color: #7ee0a1; }

@media (max-height: 520px) {
  .cam-hint { font-size: 0.85rem; padding: 0.4rem 0.7rem; }
  .cam-shutter { width: 60px; height: 60px; }
  .cam-shutter span { width: 46px; height: 46px; }
}
