/* components.css — 公共 UI 组件（所有值引用 design-tokens.css） */
/* 最后更新: 2026-04-03 */

/* 按钮 */
.btn-primary,
.btn-secondary,
.btn-danger,
.btn-ghost,
.btn-cart {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: var(--space-sm);
  min-height: calc(var(--space-xl) + var(--space-md));
  padding: calc(var(--space-sm) + var(--space-xs)) var(--space-lg);
  border: none;
  border-radius: var(--radius-full);
  font-family: var(--font-label);
  font-size: var(--text-body);
  font-weight: 600;
  line-height: 1;
  text-decoration: none;
  cursor: pointer;
  transition:
    transform var(--transition-fast),
    box-shadow var(--transition-normal),
    background var(--transition-normal),
    color var(--transition-normal),
    opacity var(--transition-fast);
}

.btn-primary:hover,
.btn-secondary:hover,
.btn-danger:hover,
.btn-ghost:hover,
.btn-cart:hover {
  transform: translateY(calc(var(--space-xs) * -1));
}

.btn-primary:active,
.btn-secondary:active,
.btn-danger:active,
.btn-ghost:active,
.btn-cart:active {
  transform: scale(0.95);
}

.btn-primary:disabled,
.btn-secondary:disabled,
.btn-danger:disabled,
.btn-ghost:disabled,
.btn-cart:disabled,
.btn-primary.is-disabled,
.btn-secondary.is-disabled,
.btn-danger.is-disabled,
.btn-ghost.is-disabled,
.btn-cart.is-disabled {
  opacity: 0.5;
  cursor: not-allowed;
  pointer-events: none;
  box-shadow: none;
  transform: none;
}

.btn-primary {
  background: linear-gradient(135deg, var(--color-primary), var(--color-primary-dim));
  color: var(--color-surface-container-lowest);
  box-shadow: var(--shadow-sm);
}

.btn-primary:hover {
  background: linear-gradient(135deg, var(--color-primary-dim), var(--color-primary));
  box-shadow: var(--shadow-md);
}

.btn-secondary {
  background: var(--color-secondary-container);
  color: var(--color-on-secondary-container);
  box-shadow: var(--shadow-sm);
}

.btn-secondary:hover {
  background: #f5d5ea;
  background: color-mix(in srgb, var(--color-secondary-container) 82%, var(--color-secondary));
  box-shadow: var(--shadow-md);
}

.btn-danger {
  background: var(--color-error);
  color: var(--color-on-error);
  box-shadow: var(--shadow-sm);
}

.btn-danger:hover {
  background: var(--color-error-dim);
  box-shadow: var(--shadow-md);
}

.btn-ghost {
  background: transparent;
  color: var(--color-primary);
  box-shadow: none;
}

.btn-ghost:hover {
  background: var(--color-surface-container-low);
  box-shadow: none;
}

.btn-cart {
  background: linear-gradient(135deg, var(--color-secondary-container), var(--color-secondary));
  color: var(--color-surface-container-lowest);
  box-shadow: var(--shadow-sm);
}

.btn-cart:hover {
  background: linear-gradient(135deg, var(--color-secondary), var(--color-secondary-dim));
  box-shadow: var(--shadow-md);
}

/* 卡片 */
.card,
.card-hero,
.card-metric {
  background: var(--color-surface-container-lowest);
  box-shadow: var(--shadow-sm);
  transition:
    transform var(--transition-normal),
    box-shadow var(--transition-normal),
    background var(--transition-normal);
}

.card,
.card-metric {
  border-radius: var(--radius-md);
}

.card-hero {
  border-radius: var(--radius-lg);
}

.card:hover,
.card-hero:hover,
.card-metric:hover {
  transform: translateY(calc(var(--space-xs) * -1));
  box-shadow: var(--shadow-md);
}

.card-metric {
  display: grid;
  place-items: center;
  gap: var(--space-sm);
  padding: var(--space-lg);
  text-align: center;
  font-family: var(--font-headline);
  font-size: var(--text-display);
  font-weight: 800;
  color: var(--color-on-surface);
}

.card-metric strong,
.card-metric .metric-value {
  font-family: var(--font-headline);
  font-size: var(--text-display);
  font-weight: 800;
  line-height: 1.1;
  color: var(--color-primary);
}

/* 输入框 */
.input,
textarea.input {
  width: 100%;
  padding: calc(var(--space-sm) + var(--space-xs)) var(--space-md);
  border: none;
  border-radius: calc(var(--radius-sm) + var(--space-xs));
  background: var(--color-surface-container-highest);
  color: var(--color-on-surface);
  font-family: var(--font-body);
  font-size: var(--text-body);
  line-height: 1.6;
  transition:
    background var(--transition-normal),
    box-shadow var(--transition-normal),
    transform var(--transition-fast);
}

.input::placeholder,
textarea.input::placeholder {
  color: var(--color-on-surface-variant);
}

.input:focus,
textarea.input:focus {
  outline: none;
  background: var(--color-surface-container-lowest);
  box-shadow:
    inset 0 0 0 2px rgba(106, 55, 212, 0.28),
    var(--shadow-glow);
  box-shadow:
    inset 0 0 0 2px color-mix(in srgb, var(--color-primary) 28%, transparent),
    var(--shadow-glow);
}

.input-search {
  padding-left: calc(var(--space-xl) + var(--space-sm));
  border-radius: var(--radius-full);
}

textarea.input {
  min-height: calc(var(--space-2xl) * 2);
  resize: vertical;
}

/* 导航 */
.glass-nav {
  background: var(--glass-bg);
  backdrop-filter: var(--glass-blur);
  -webkit-backdrop-filter: var(--glass-blur);
  box-shadow: var(--shadow-ethereal);
}

.nav-link {
  position: relative;
  display: inline-flex;
  align-items: center;
  gap: var(--space-xs);
  padding: var(--space-sm) 0;
  color: var(--color-on-surface-variant);
  font-family: var(--font-label);
  font-size: var(--text-body);
  font-weight: 500;
  text-decoration: none;
  transition: color var(--transition-fast);
}

.nav-link::after {
  content: "";
  position: absolute;
  left: 0;
  right: 0;
  bottom: calc(var(--space-xs) * -1);
  height: 2px;
  border-radius: var(--radius-full);
  background: var(--color-primary);
  transform: scaleX(0);
  transform-origin: center;
  transition: transform var(--transition-normal);
}

.nav-link:hover,
.nav-link.is-active,
.nav-link[aria-current="page"] {
  color: var(--color-primary);
}

.nav-link.is-active::after,
.nav-link[aria-current="page"]::after {
  transform: scaleX(1);
}

.bottom-nav {
  position: fixed;
  right: 0;
  bottom: 0;
  left: 0;
  z-index: 40;
  display: flex;
  align-items: center;
  justify-content: space-around;
  gap: var(--space-sm);
  padding: var(--space-sm) var(--space-md) calc(var(--space-md) + env(safe-area-inset-bottom));
  background: var(--glass-bg);
  backdrop-filter: var(--glass-blur);
  -webkit-backdrop-filter: var(--glass-blur);
  box-shadow: var(--shadow-ethereal);
}

/* 徽标 */
.badge {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  gap: var(--space-xs);
  padding: 2px var(--space-sm);
  border-radius: calc(var(--radius-sm) + var(--space-xs));
  font-family: var(--font-label);
  font-size: var(--text-badge);
  font-weight: 600;
  line-height: 1.2;
  white-space: nowrap;
}

.badge-primary {
  background: #f3e8ff;
  background: color-mix(in srgb, var(--color-primary) 10%, transparent);
  color: var(--color-primary);
}

.badge-success {
  background: var(--color-success-bg);
  color: var(--color-success);
}

.badge-warning {
  background: var(--color-warning-bg);
  color: var(--color-warning);
}

.badge-error {
  background: var(--color-error-container);
  color: var(--color-on-error-container);
}

.badge-cart {
  position: absolute;
  top: calc(var(--space-xs) * -1);
  right: calc(var(--space-xs) * -1);
  min-width: calc(var(--space-md) + var(--space-xs));
  min-height: calc(var(--space-md) + var(--space-xs));
  padding: 2px calc(var(--space-xs) + 2px);
  border-radius: var(--radius-full);
  background: var(--color-secondary);
  color: var(--color-surface-container-lowest);
  box-shadow: var(--shadow-sm);
}

/* 表格 */
.table-ethereal {
  width: 100%;
  border-collapse: collapse;
  background: var(--color-surface-container-lowest);
  color: var(--color-on-surface);
  font-family: var(--font-body);
}

.table-ethereal th,
.table-ethereal td {
  padding: var(--space-md);
  border: none;
  text-align: left;
}

.table-ethereal thead th {
  background: var(--color-surface-container-low);
  color: var(--color-on-surface-variant);
  font-family: var(--font-label);
  font-size: var(--text-caption);
  font-weight: 600;
}

.table-ethereal tbody tr {
  transition: background var(--transition-fast);
}

.table-ethereal tbody tr:hover {
  background: #f5f0ff;
  background: color-mix(in srgb, var(--color-primary) 8%, var(--color-surface-container-lowest));
}

/* Flash 消息 */
.flash-success,
.flash.success,
.flash-error,
.flash.error,
.flash-warning,
.flash.warning {
  padding: var(--space-md) var(--space-lg);
  border-radius: var(--radius-md);
  font-family: var(--font-body);
  font-size: var(--text-body);
  font-weight: 500;
  box-shadow: var(--shadow-sm);
}

.flash-success,
.flash.success {
  background: var(--color-success-bg);
  color: var(--color-success);
}

.flash-error,
.flash.error {
  background: #fef2f2;
  background: color-mix(in srgb, var(--color-error) 12%, var(--color-surface-container-lowest));
  color: var(--color-error);
}

.flash-warning,
.flash.warning {
  background: var(--color-warning-bg);
  color: var(--color-warning);
}

/* 弹窗与抽屉 */
.drawer {
  position: fixed;
  right: 0;
  bottom: 0;
  left: 0;
  z-index: 60;
  padding: var(--space-lg);
  border-radius: var(--radius-lg) var(--radius-lg) 0 0;
  background: var(--glass-bg);
  backdrop-filter: var(--glass-blur);
  -webkit-backdrop-filter: var(--glass-blur);
  box-shadow: var(--shadow-lg);
  transform: translateY(100%);
  transition: transform var(--transition-slow);
}

.drawer.is-open {
  transform: translateY(0);
}

@media (min-width: 768px) {
  .drawer {
    top: 0;
    bottom: 0;
    left: auto;
    width: min(32rem, 100%);
    border-radius: var(--radius-lg) 0 0 var(--radius-lg);
    transform: translateX(100%);
  }

  .drawer.is-open {
    transform: translateX(0);
  }
}

.modal-overlay {
  position: fixed;
  inset: 0;
  z-index: 50;
  background: rgba(14, 13, 20, 0.48);
  background: color-mix(in srgb, var(--color-inverse-surface) 48%, transparent);
}

/* 工具类 */
.text-price {
  color: var(--color-price);
  font-weight: 700;
}

.text-muted {
  color: var(--color-on-surface-variant);
}

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

/* 字号工具类 */
.text-h1 { font-size: var(--text-h1); }
.text-h2 { font-size: var(--text-h2); }
.text-display { font-size: var(--text-display); }
.text-body { font-size: var(--text-body); }
.text-caption { font-size: var(--text-caption); }

/* 字体工具类 */
.font-headline { font-family: var(--font-headline); }
.font-body { font-family: var(--font-body); }
.font-label { font-family: var(--font-label); }

/* 分页导航 */
.shop-pagination {
  display: flex;
  gap: var(--space-sm);
  align-items: center;
  flex-wrap: wrap;
  justify-content: flex-end;
  margin-top: var(--space-md);
}

.shop-pagination-link {
  padding: var(--space-sm) var(--space-md);
  border-radius: var(--radius-sm);
  text-decoration: none;
  color: var(--color-on-surface);
  background: var(--color-surface-container-lowest);
  transition: background var(--transition-fast), color var(--transition-fast);
}

.shop-pagination-link:hover {
  background: var(--color-surface-container-low);
}

.shop-pagination-link--active {
  background: var(--color-primary);
  color: #fff;
}

.shop-pagination-link--active:hover {
  background: var(--color-primary);
}

/* ═══════════════════════════════════════════════════════════════
 * 现代电商组件库（v1.12.x+，v1+v2 共享）
 * 用于 2.0 UI 升级前的基础设施准备。
 * 设计原则：纯 CSS + design-tokens 变量，零外部依赖（铁律 #3）。
 * ═══════════════════════════════════════════════════════════════ */

/* ── Toast 通知 ─────────────────────────────────────────────────
 * 替代顶部 flash 的瞬态提示。3-5 秒自动消失（JS 层控制），
 * 多条堆叠时按时间倒序在右下角往上推。
 * 用法：
 *   <div class="toast toast--success">已加入收藏</div>
 *   <div class="toast toast--error">操作失败</div>
 */
.toast-container {
  position: fixed;
  bottom: var(--space-lg, 24px);
  right: var(--space-lg, 24px);
  z-index: 9999;
  display: flex;
  flex-direction: column-reverse;
  gap: var(--space-sm, 8px);
  pointer-events: none;
}

.toast {
  pointer-events: auto;
  min-width: 200px;
  max-width: 360px;
  padding: var(--space-sm, 8px) var(--space-md, 16px);
  border-radius: var(--radius-md, 16px);
  background: var(--color-surface-container-lowest, #fff);
  color: var(--color-on-surface, #2f2e35);
  box-shadow: var(--shadow-lg, 0 8px 24px rgba(106, 55, 212, 0.12));
  font-size: var(--text-body, 0.9375rem);
  display: flex;
  align-items: center;
  gap: var(--space-xs, 4px);
  animation: toast-slide-in 240ms ease-out;
}

.toast--success { border-left: 3px solid var(--color-success, #34d399); }
.toast--error   { border-left: 3px solid var(--color-error, #b41340); }
.toast--warning { border-left: 3px solid var(--color-warning, #fbbf24); }
.toast--info    { border-left: 3px solid var(--color-info, #60a5fa); }

@keyframes toast-slide-in {
  from { transform: translateX(100%); opacity: 0; }
  to   { transform: translateX(0); opacity: 1; }
}

@keyframes toast-fade-out {
  to { transform: translateX(100%); opacity: 0; }
}

.toast--leaving { animation: toast-fade-out 240ms ease-in forwards; }

/* ── Modal / Drawer ─────────────────────────────────────────────
 * 居中弹窗 + 底部抽屉两种姿态。
 * 用法：
 *   <div class="modal-backdrop is-open">
 *     <div class="modal-card">...内容...</div>
 *   </div>
 */
.modal-backdrop {
  position: fixed;
  inset: 0;
  background: rgba(0, 0, 0, 0.5);
  backdrop-filter: blur(4px);
  z-index: 1000;
  display: none;
  align-items: center;
  justify-content: center;
  padding: var(--space-md, 16px);
  animation: modal-fade-in 200ms ease-out;
}

.modal-backdrop.is-open { display: flex; }

.modal-card {
  background: var(--color-surface-container-lowest, #fff);
  border-radius: var(--radius-lg, 2rem);
  max-width: 480px;
  width: 100%;
  max-height: 90vh;
  overflow-y: auto;
  padding: var(--space-lg, 24px);
  box-shadow: var(--shadow-lg, 0 20px 60px rgba(106, 55, 212, 0.18));
  animation: modal-pop-in 240ms cubic-bezier(0.34, 1.56, 0.64, 1);
}

/* Drawer：从底部滑入（移动端友好）*/
.modal-backdrop--drawer { align-items: flex-end; }
.modal-backdrop--drawer .modal-card {
  border-radius: var(--radius-lg, 2rem) var(--radius-lg, 2rem) 0 0;
  max-width: none;
  margin-bottom: 0;
  animation: drawer-slide-up 280ms ease-out;
}

@keyframes modal-fade-in { from { opacity: 0; } to { opacity: 1; } }
@keyframes modal-pop-in {
  from { transform: scale(0.94); opacity: 0; }
  to   { transform: scale(1); opacity: 1; }
}
@keyframes drawer-slide-up {
  from { transform: translateY(100%); }
  to   { transform: translateY(0); }
}

/* ── Sticky 底部操作栏 ───────────────────────────────────────────
 * 淘宝风格：详情页 / 结算页 fixed 在底部，不被键盘遮挡（用户输入时上推）。
 * 配合 main 元素 padding-bottom 防被遮。
 * 用法：
 *   <div class="sticky-bar"><div class="sticky-bar-inner">...</div></div>
 */
.sticky-bar {
  position: fixed;
  left: 0;
  right: 0;
  bottom: 0;
  z-index: 100;
  background: var(--color-surface-container-lowest, #fff);
  border-top: 1px solid var(--color-surface-container, #ebe6f2);
  padding: var(--space-sm, 8px) var(--space-md, 16px);
  padding-bottom: calc(var(--space-sm, 8px) + env(safe-area-inset-bottom, 0));
  box-shadow: 0 -4px 16px rgba(106, 55, 212, 0.06);
}

.sticky-bar-inner {
  max-width: 1200px;
  margin: 0 auto;
  display: flex;
  gap: var(--space-sm, 8px);
  align-items: center;
}

/* ── Skeleton 加载占位 ──────────────────────────────────────────
 * 渐变 shimmer 动画。
 * 用法：<div class="skeleton skeleton--text"></div>
 */
.skeleton {
  background: linear-gradient(
    90deg,
    var(--color-surface-container-low, #f4effa) 0%,
    var(--color-surface-container, #ebe6f2) 50%,
    var(--color-surface-container-low, #f4effa) 100%
  );
  background-size: 200% 100%;
  animation: skeleton-shimmer 1.4s ease-in-out infinite;
  border-radius: var(--radius-sm, 0.5rem);
}

.skeleton--text   { height: 1em; margin: 0.25em 0; }
.skeleton--title  { height: 1.5em; width: 70%; }
.skeleton--avatar { width: 48px; height: 48px; border-radius: 50%; }
.skeleton--card   { height: 200px; }

@keyframes skeleton-shimmer {
  0%   { background-position: 200% 0; }
  100% { background-position: -200% 0; }
}

/* ── Empty State 空态 ──────────────────────────────────────────
 * 用法：
 *   <div class="empty-state">
 *     <span class="material-symbols-outlined empty-state-icon">inbox</span>
 *     <h3 class="empty-state-title">暂无数据</h3>
 *     <p class="empty-state-desc">说明文字</p>
 *     <a class="btn-primary">主操作</a>
 *   </div>
 */
.empty-state {
  text-align: center;
  padding: var(--space-2xl, 48px) var(--space-md, 16px);
  color: var(--color-outline, #78757e);
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: var(--space-md, 16px);
}

.empty-state-icon {
  font-size: 3rem;
  opacity: 0.4;
}

.empty-state-title {
  font-size: var(--text-h2, 1.25rem);
  font-weight: 600;
  color: var(--color-on-surface, #2f2e35);
  margin: 0;
}

.empty-state-desc {
  font-size: var(--text-body, 0.9375rem);
  margin: 0;
  max-width: 320px;
}

/* ── 触摸目标尺寸（a11y 合规）────────────────────────────────────
 * 移动端任何 button / a / [role="button"] 最小 44×44，
 * 满足 WCAG 2.2 Target Size + iOS HIG。
 */
@media (hover: none) and (pointer: coarse) {
  button:not(.btn-sm),
  a.btn-primary,
  a.btn-secondary,
  a.btn-ghost,
  [role="button"] {
    min-height: 44px;
    min-width: 44px;
  }
}

/* ── 焦点可见（键盘导航）──────────────────────────────────────────
 * 默认 :focus-visible 浏览器风格不一致，统一加紫色 outline。
 */
*:focus-visible {
  outline: 2px solid var(--color-primary, #6a37d4);
  outline-offset: 2px;
}

/* ═══════════════════════════════════════════════════════════════
 * Utility classes（v1.14+，ARCH-3 基础设施）
 *
 * 用途：替代散落在模板的 inline style="..."，减少违反 CSS 铁律 #11-13
 * 的次数。设计原则：
 *   - 命名参考 Tailwind 但更精简（仅 cover 项目实际用到的高频模式）
 *   - 单位用 design-tokens 变量保持视觉一致
 *   - 不引入外部依赖（铁律 #3）
 *
 * 渐进式重构策略：v1 模板 393 处 inline style 不一次性替换
 * （会有视觉回归风险），随着 v2.0 UI 收敛时一起做。
 * ═══════════════════════════════════════════════════════════════ */

/* ── Flex / Grid ───────────────────────────────────────────────── */
.u-flex            { display: flex; }
.u-flex-col        { display: flex; flex-direction: column; }
.u-inline-flex     { display: inline-flex; }
.u-grid            { display: grid; }
.u-block           { display: block; }
.u-inline-block    { display: inline-block; }
.u-hidden          { display: none; }

.u-items-center    { align-items: center; }
.u-items-start     { align-items: flex-start; }
.u-items-end       { align-items: flex-end; }
.u-justify-center  { justify-content: center; }
.u-justify-between { justify-content: space-between; }
.u-justify-end     { justify-content: flex-end; }
.u-flex-wrap       { flex-wrap: wrap; }
.u-flex-1          { flex: 1; }
.u-flex-shrink-0   { flex-shrink: 0; }

/* gap */
.u-gap-xs   { gap: var(--space-xs, 4px); }
.u-gap-sm   { gap: var(--space-sm, 8px); }
.u-gap-md   { gap: var(--space-md, 16px); }
.u-gap-lg   { gap: var(--space-lg, 24px); }

/* ── Spacing ───────────────────────────────────────────────────── */
.u-m-0      { margin: 0; }
.u-mt-xs    { margin-top: var(--space-xs, 4px); }
.u-mt-sm    { margin-top: var(--space-sm, 8px); }
.u-mt-md    { margin-top: var(--space-md, 16px); }
.u-mt-lg    { margin-top: var(--space-lg, 24px); }
.u-mt-xl    { margin-top: var(--space-xl, 32px); }
.u-mb-xs    { margin-bottom: var(--space-xs, 4px); }
.u-mb-sm    { margin-bottom: var(--space-sm, 8px); }
.u-mb-md    { margin-bottom: var(--space-md, 16px); }
.u-mb-lg    { margin-bottom: var(--space-lg, 24px); }
.u-mx-auto  { margin-left: auto; margin-right: auto; }
.u-p-sm     { padding: var(--space-sm, 8px); }
.u-p-md     { padding: var(--space-md, 16px); }
.u-p-lg     { padding: var(--space-lg, 24px); }
.u-px-md    { padding-left: var(--space-md, 16px); padding-right: var(--space-md, 16px); }
.u-py-md    { padding-top: var(--space-md, 16px); padding-bottom: var(--space-md, 16px); }

/* ── Typography ────────────────────────────────────────────────── */
.u-text-caption { font-size: var(--text-caption, 0.8125rem); }
.u-text-body    { font-size: var(--text-body, 0.9375rem); }
.u-text-h2      { font-size: var(--text-h2, 1.25rem); }
.u-text-h1      { font-size: var(--text-h1, 1.625rem); }
.u-text-muted   { color: var(--color-outline, #78757e); }
.u-text-on-surface { color: var(--color-on-surface, #2f2e35); }
.u-text-primary { color: var(--color-primary, #6a37d4); }
.u-text-error   { color: var(--color-error, #b41340); }
.u-text-success { color: var(--color-success, #34d399); }
.u-text-warning { color: var(--color-warning, #fbbf24); }
.u-text-white   { color: #fff; }
.u-text-center  { text-align: center; }
.u-text-left    { text-align: left; }
.u-text-right   { text-align: right; }
.u-font-bold    { font-weight: 700; }
.u-font-medium  { font-weight: 500; }
.u-truncate     { overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.u-no-decoration { text-decoration: none; }

/* ── Sizing ────────────────────────────────────────────────────── */
.u-w-full       { width: 100%; }
.u-h-full       { height: 100%; }
.u-aspect-square { aspect-ratio: 1; }

/* ── Background / Border ───────────────────────────────────────── */
.u-bg-surface       { background: var(--color-surface, #faf5ff); }
.u-bg-surface-low   { background: var(--color-surface-container-low, #f4effa); }
.u-bg-surface-lowest { background: var(--color-surface-container-lowest, #fff); }
.u-bg-primary       { background: var(--color-primary, #6a37d4); }
.u-bg-error         { background: var(--color-error, #b41340); }
.u-bg-success       { background: var(--color-success, #34d399); }
.u-rounded-sm       { border-radius: var(--radius-sm, 0.5rem); }
.u-rounded-md       { border-radius: var(--radius-md, 1rem); }
.u-rounded-lg       { border-radius: var(--radius-lg, 2rem); }
.u-rounded-full     { border-radius: 9999px; }
.u-object-cover     { object-fit: cover; }
.u-overflow-hidden  { overflow: hidden; }

/* ── Position ──────────────────────────────────────────────────── */
.u-relative { position: relative; }
.u-absolute { position: absolute; }
.u-fixed    { position: fixed; }

/* ── Cursor ────────────────────────────────────────────────────── */
.u-cursor-pointer { cursor: pointer; }
.u-cursor-default { cursor: default; }

