/* ─── Radio + Radiobutton — shared component styles ──────────────────────────
   Single source of truth for component CSS.
   Values: site/tokens.css (--ds-radio-* tokens)
   Consumers: site/preview/radio.html + src/components/Radio/Radio.tsx
              + src/components/Radio/Radiobutton.tsx
   Naming: ds- prefix replaces CSS Modules scoping
   States: data-attributes only (data-selected / data-hovered / data-pressed /
           data-disabled / data-error / data-focus-visible / data-size / data-variant)
   ─────────────────────────────────────────────────────────────────────────── */

/* ── Radio root ────────────────────────────────────────────────────────────── */

.ds-radio {
  display: inline-flex;
  /* Контрол top-aligned: круг встаёт у ПЕРВОЙ строки лейбла (Figma 1129:2635),
     а не центрируется по высоте «лейбл + описание» при переносе/описании. */
  align-items: flex-start;
  gap: var(--ds-radio-md-gap);
  cursor: pointer;
  /* Touch-target 40px (MD) сохраняется через симметричный вертикальный паддинг
     (touch − circle)/2 вместо min-height: при одной строке круг визуально по центру
     40px-зоны, при переносе — top-aligned. Токен-производное, без хардкода. */
  padding-top: calc((var(--ds-radio-md-touch) - var(--ds-radio-md-circle)) / 2);
  padding-bottom: calc((var(--ds-radio-md-touch) - var(--ds-radio-md-circle)) / 2);
  position: relative;
  user-select: none;
}

.ds-radio[data-size='sm'] {
  gap: var(--ds-radio-sm-gap);
  /* SM touch-target 36px: (36 − 20)/2 */
  padding-top: calc((var(--ds-radio-sm-touch) - var(--ds-radio-sm-circle)) / 2);
  padding-bottom: calc((var(--ds-radio-sm-touch) - var(--ds-radio-sm-circle)) / 2);
}

.ds-radio[data-disabled] {
  cursor: not-allowed;
}

/* ── Visually hidden native input ──────────────────────────────────────────── */

.ds-radio-input {
  position: absolute;
  width: 1px;
  height: 1px;
  padding: 0;
  margin: -1px;
  overflow: hidden;
  clip: rect(0, 0, 0, 0);
  white-space: nowrap;
  border: 0;
}

/* ── Circle control — shared between Radio and Radiobutton ─────────────────── */

.ds-radio-ctrl {
  position: relative;
  display: flex;
  align-items: center;
  justify-content: center;
  flex-shrink: 0;
  width: var(--ds-radio-md-circle);
  height: var(--ds-radio-md-circle);
  border-radius: 999px;
  border: var(--ds-radio-stroke) solid var(--ds-nba-border);
  background: var(--ds-surface);
  transition:
    border-color var(--ds-dur-2) var(--ds-ease),
    background   var(--ds-dur-2) var(--ds-ease),
    box-shadow   var(--ds-dur-2) var(--ds-ease);
}

:is(.ds-radio, .ds-rb)[data-size='sm'] .ds-radio-ctrl {
  width: var(--ds-radio-sm-circle);
  height: var(--ds-radio-sm-circle);
}

/* ── Circle states ─────────────────────────────────────────────────────────── */

:is(.ds-radio, .ds-rb)[data-hovered] .ds-radio-ctrl,
:is(.ds-radio, .ds-rb):hover:not([data-disabled]) .ds-radio-ctrl {
  border-color: var(--ds-stroke-neutral-hover);
  background: var(--ds-nba-20);
}

:is(.ds-radio, .ds-rb)[data-pressed] .ds-radio-ctrl {
  border-color: var(--ds-stroke-neutral-hover);
  background: var(--ds-nba-40);
}

:is(.ds-radio, .ds-rb)[data-selected] .ds-radio-ctrl {
  border: 0;
  background: var(--ds-neutral-fill);
}

:is(.ds-radio, .ds-rb)[data-selected][data-hovered] .ds-radio-ctrl,
:is(.ds-radio, .ds-rb)[data-selected]:hover:not([data-disabled]) .ds-radio-ctrl {
  border: 0;
  background: var(--ds-neutral-fill-hover);
}

:is(.ds-radio, .ds-rb)[data-selected][data-pressed] .ds-radio-ctrl {
  border: 0;
  background: var(--ds-neutral-fill-pressed);
}

:is(.ds-radio, .ds-rb)[data-focus-visible] .ds-radio-ctrl {
  outline: var(--ds-focus-width) solid var(--ds-info-800);
  outline-offset: var(--ds-focus-offset);
}

:is(.ds-radio, .ds-rb)[data-disabled] .ds-radio-ctrl {
  border-color: var(--ds-nba-border-disabled);
  background: var(--ds-surface);
}

:is(.ds-radio, .ds-rb)[data-selected][data-disabled] .ds-radio-ctrl {
  border: 0;
  background: var(--ds-nba-15);
}

/* Disabled foreground — Figma token-модель (БЕЗ opacity:0.4): каждый элемент
   красится своим disabled-токеном, текст остаётся читаемым (read-only-семантика).
   --ds-disabled-text спарен с nba-15 fill по своему определению. Тёмная тема —
   автоматически через значения токенов, доп. override не нужен. */
:is(.ds-radio, .ds-rb)[data-disabled] .ds-radio-dot {
  background: var(--ds-disabled-text);
}

:is(.ds-radio, .ds-rb)[data-disabled]
  :is(.ds-radio-label, .ds-radio-desc, .ds-rb-label, .ds-rb-caption) {
  color: var(--ds-disabled-text);
}

.ds-rb[data-disabled] .ds-rb-icon-help {
  color: var(--ds-disabled-text);
}

/* ── Error states ──────────────────────────────────────────────────────────── */

:is(.ds-radio, .ds-rb)[data-error] .ds-radio-ctrl {
  border-width: var(--ds-radio-stroke-negative);
  border-color: var(--ds-neg-default);
  background: var(--ds-surface);
}

:is(.ds-radio, .ds-rb)[data-error][data-hovered] .ds-radio-ctrl,
:is(.ds-radio, .ds-rb)[data-error]:hover:not([data-disabled]) .ds-radio-ctrl {
  border-width: var(--ds-radio-stroke-negative);
  border-color: var(--ds-neg-hover);
  background: var(--ds-nba-20);
}

:is(.ds-radio, .ds-rb)[data-error][data-selected] .ds-radio-ctrl {
  border: 0;
  background: var(--ds-neg-default);
}

:is(.ds-radio, .ds-rb)[data-error][data-selected][data-hovered] .ds-radio-ctrl,
:is(.ds-radio, .ds-rb)[data-error][data-selected]:hover:not([data-disabled]) .ds-radio-ctrl {
  border: 0;
  background: var(--ds-neg-hover);
}

/* ── Inner dot ─────────────────────────────────────────────────────────────── */

.ds-radio-dot {
  display: block;
  width: var(--ds-radio-md-dot);
  height: var(--ds-radio-md-dot);
  border-radius: 999px;
  background: var(--ds-text-invert);
  opacity: 0;
  transform: scale(0);
  transition:
    opacity   var(--ds-dur-2) var(--ds-ease),
    transform var(--ds-dur-1) var(--ds-ease);
}

:is(.ds-radio, .ds-rb)[data-size='sm'] .ds-radio-dot {
  width: var(--ds-radio-sm-dot);
  height: var(--ds-radio-sm-dot);
}

:is(.ds-radio, .ds-rb)[data-selected] .ds-radio-dot {
  opacity: 1;
  transform: scale(1);
}

/* ── Radio label area ──────────────────────────────────────────────────────── */

.ds-radio-txt {
  display: flex;
  flex-direction: column;
  overflow: hidden;
}

.ds-radio-label {
  font-size: var(--ds-radio-md-fs);
  line-height: var(--ds-radio-md-lh);
  color: var(--ds-text-primary);
}

.ds-radio[data-size='sm'] .ds-radio-label {
  font-size: var(--ds-radio-sm-fs);
  line-height: var(--ds-radio-sm-lh);
}

.ds-radio-desc {
  font-size: var(--ds-radio-md-fs);
  line-height: var(--ds-radio-md-lh);
  color: var(--ds-text-secondary);
}

.ds-radio[data-size='sm'] .ds-radio-desc {
  font-size: var(--ds-radio-sm-fs);
  line-height: var(--ds-radio-sm-lh);
}

/* ── Radiobutton root ──────────────────────────────────────────────────────── */

.ds-rb {
  display: inline-flex;
  align-items: flex-start;
  cursor: pointer;
  user-select: none;
  position: relative;
  border-radius: var(--ds-radio-cont-r-md);
  transition: background var(--ds-dur-2) var(--ds-ease);
}

.ds-rb[data-size='sm'] {
  border-radius: var(--ds-radio-cont-r-sm);
}

.ds-rb[data-disabled] {
  cursor: not-allowed;
}

/* Container variant */
.ds-rb[data-variant='container'] {
  padding: var(--ds-radio-cont-py-md) var(--ds-radio-cont-px-md);
}

.ds-rb[data-variant='container'][data-size='sm'] {
  padding: var(--ds-radio-cont-py-sm) var(--ds-radio-cont-px-sm);
}

.ds-rb[data-variant='container'][data-hovered] {
  background: var(--ds-nba-20);
}

.ds-rb[data-variant='container'][data-pressed] {
  background: var(--ds-nba-40);
}

/* Container keeps ctrl background as surface (container bg carries the hover) */
.ds-rb[data-variant='container'][data-hovered] .ds-radio-ctrl,
.ds-rb[data-variant='container'][data-pressed] .ds-radio-ctrl {
  background: var(--ds-surface);
}

.ds-rb[data-variant='container'][data-error][data-hovered] .ds-radio-ctrl {
  background: var(--ds-surface);
}

/* ── Hit area — клик-зона выбора (radio + label + caption), БЕЗ Help Icon ──────
   Help Icon вынесена из <label> в отдельный <button>, поэтому клик по ней
   не выбирает Radio. ::before растягивает клик-зону на всю карточку (включая
   padding контейнера), а Help-кнопка лежит выше overlay по z-index. */

.ds-rb-hit {
  display: flex;
  align-items: flex-start;
  flex: 1 1 auto;
  min-width: 0;
  cursor: pointer;
}

.ds-rb-hit::before {
  content: '';
  position: absolute;
  inset: 0;
  border-radius: inherit;
}

.ds-rb[data-disabled] .ds-rb-hit {
  cursor: not-allowed;
}

/* ── Radiobutton inner elements ────────────────────────────────────────────── */

.ds-rb-core {
  display: flex;
  flex-direction: column;
  gap: var(--ds-radio-core-gap);
  flex: 1 0 0;
  min-width: 0;
}

.ds-rb-label-row {
  display: flex;
  align-items: center;
  gap: var(--ds-radio-label-gap-md);
  padding: 0 var(--ds-radio-label-px-md);
}

.ds-rb[data-size='sm'] .ds-rb-label-row {
  gap: var(--ds-radio-label-gap-sm);
  padding: 0 var(--ds-radio-label-px-sm);
}

.ds-rb-label {
  flex: 1 0 0;
  min-width: 0;
  font-size: var(--ds-radio-md-fs);
  line-height: var(--ds-radio-md-lh);
  font-weight: 500;
  color: var(--ds-text-primary);
}

.ds-rb[data-size='sm'] .ds-rb-label {
  font-size: var(--ds-radio-sm-fs);
  line-height: var(--ds-radio-sm-lh);
}

/* Help — интерактивный триггер ВНЕ клик-зоны. Wrapper держит z-index выше
   .ds-rb-hit::before, чтобы клик/hover по Help не доходил до выбора Radio.
   (Tooltip рендерит .ds-tooltip-root; добавочный класс не конфликтует.) */
.ds-rb-help-tip {
  position: relative;
  z-index: 1;
  flex-shrink: 0;
  display: inline-flex;
  align-items: flex-start;
}

.ds-rb-icon-help {
  display: flex;
  align-items: flex-start;
  padding: var(--ds-radio-icon-py-md) 0;
  flex-shrink: 0;
  color: var(--ds-text-secondary);
  background: transparent;
  border: 0;
  margin: 0;
  cursor: pointer;
  appearance: none;
  -webkit-appearance: none;
  border-radius: 999px;
}

.ds-rb[data-size='sm'] .ds-rb-icon-help {
  padding: var(--ds-radio-icon-py-sm) 0;
}

.ds-rb-icon-help:focus-visible {
  outline: none;
  box-shadow: var(--ds-shadow-focus);
}

.ds-rb-caption {
  display: block;
  padding: 0 var(--ds-radio-caption-px-md);
  font-size: var(--ds-radio-caption-fs);
  line-height: var(--ds-radio-caption-lh);
  font-weight: 400;
  color: var(--ds-text-secondary);
}

.ds-rb[data-size='sm'] .ds-rb-caption {
  padding: 0 var(--ds-radio-caption-px-sm);
}

/* ── Dark mode — hover bg elevated from 20% to 60% alpha ───────────────── */

[data-theme="dark"] :is(.ds-radio, .ds-rb)[data-hovered] .ds-radio-ctrl,
[data-theme="dark"] :is(.ds-radio, .ds-rb):hover:not([data-disabled]) .ds-radio-ctrl {
  background: var(--ds-nba-40);
}

[data-theme="dark"] :is(.ds-radio, .ds-rb)[data-error][data-hovered] .ds-radio-ctrl,
[data-theme="dark"] :is(.ds-radio, .ds-rb)[data-error]:hover:not([data-disabled]) .ds-radio-ctrl {
  background: var(--ds-nba-40);
}

[data-theme="dark"] .ds-rb[data-variant='container'][data-hovered] {
  background: var(--ds-nba-40);
}

@media (prefers-color-scheme: dark) {
  :root:not([data-theme="light"]) :is(.ds-radio, .ds-rb)[data-hovered] .ds-radio-ctrl,
  :root:not([data-theme="light"]) :is(.ds-radio, .ds-rb):hover:not([data-disabled]) .ds-radio-ctrl {
    background: var(--ds-nba-40);
  }

  :root:not([data-theme="light"]) :is(.ds-radio, .ds-rb)[data-error][data-hovered] .ds-radio-ctrl,
  :root:not([data-theme="light"]) :is(.ds-radio, .ds-rb)[data-error]:hover:not([data-disabled]) .ds-radio-ctrl {
    background: var(--ds-nba-40);
  }

  :root:not([data-theme="light"]) .ds-rb[data-variant='container'][data-hovered] {
    background: var(--ds-nba-40);
  }
}

/* ── Reduced motion ────────────────────────────────────────────────────────── */

@media (prefers-reduced-motion: reduce) {
  .ds-radio-ctrl,
  .ds-radio-dot {
    transition-duration: 0.01ms !important;
    animation-duration: 0.01ms !important;
  }
}

/* ── RadioGroup ──────────────────────────────────────────────────────────── */

.ds-radio-group {
  border: none;
  padding: 0;
  margin: 0;
  min-inline-size: 0;
}

.ds-radio-group-legend {
  font-size: var(--ds-radio-md-fs);
  line-height: var(--ds-radio-md-lh);
  color: var(--ds-text-primary);
  font-weight: 500;
  padding: 0;
  margin-bottom: var(--ds-radio-md-group-gap);
}

.ds-radio-group-legend-required {
  color: var(--ds-neg-default);
  margin-left: var(--ds-radio-sm-gap);
}

.ds-radio-group-items {
  display: flex;
  flex-direction: column;
  gap: var(--ds-radio-md-group-gap);
}

.ds-radio-group-items[data-orientation='horizontal'] {
  flex-direction: row;
  flex-wrap: wrap;
}

.ds-radio-group-items[data-size='sm'] {
  gap: var(--ds-radio-sm-group-gap);
}

.ds-radio-group-error {
  font-size: var(--ds-radio-md-fs);
  line-height: var(--ds-radio-md-lh);
  color: var(--ds-neg-default);
  margin-top: var(--ds-radio-sm-group-gap);
  margin-bottom: 0;
}

[data-theme="dark"] .ds-radio-group-legend-required,
.theme-invert .ds-radio-group-legend-required { color: var(--ds-neg-500); }
[data-theme="dark"] .ds-radio-group-error,
.theme-invert .ds-radio-group-error { color: var(--ds-neg-500); }
@media (prefers-color-scheme: dark) {
  :root:not([data-theme="light"]) .ds-radio-group-legend-required { color: var(--ds-neg-500); }
  :root:not([data-theme="light"]) .ds-radio-group-error            { color: var(--ds-neg-500); }
}
