/* ============================================================
   DTvpn 分流中台 — 业务样式  (style_id: saas-dark-web)
   所有视觉值引用 tokens.css 的 var(--...)，零字面量。
   ============================================================ */
* { box-sizing: border-box; }

body {
  margin: 0;
  font-family: var(--font-family-sans);
  font-size: var(--font-size-base);
  line-height: var(--line-height-normal);
  color: var(--color-text-primary);
  background: var(--color-app-bg);
  -webkit-font-smoothing: antialiased;
}

.hidden { display: none !important; }
.muted { color: var(--color-text-muted); }
.spacer { flex: 1; }

/* -------- 通用控件 -------- */
button {
  cursor: pointer;
  border: 0;
  border-radius: var(--radius-md);
  padding: var(--space-md) var(--space-2xl);
  min-height: var(--size-hit);
  background: var(--color-brand);
  color: var(--color-text-on-accent);
  font-family: inherit;
  font-size: var(--font-size-base);
  font-weight: var(--font-weight-medium);
  transition: background var(--motion-fast) var(--motion-ease),
              box-shadow var(--motion-fast) var(--motion-ease),
              transform var(--motion-fast) var(--motion-ease);
}
button:hover { background: var(--color-brand-strong); }
button:active { transform: translateY(1px); }
button.ghost {
  background: transparent;
  border: 1px solid var(--color-border);
  color: var(--color-text-muted);
}
button.ghost:hover {
  background: var(--color-surface-3);
  border-color: var(--color-border-strong);
  color: var(--color-text-primary);
}

input, select {
  background: var(--color-surface-2);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  padding: var(--space-md) var(--space-lg);
  min-height: var(--size-hit);
  color: var(--color-text-primary);
  font-family: inherit;
  font-size: var(--font-size-base);
  outline: none;
  transition: border-color var(--motion-fast) var(--motion-ease);
}
input::placeholder { color: var(--color-text-muted); }
input:focus, select:focus { border-color: var(--color-brand); }

/* 焦点可见 (A11y) */
:focus-visible {
  outline: var(--focus-width) solid var(--color-focus-ring);
  outline-offset: var(--focus-width);
}

.err {
  color: var(--color-danger);
  min-height: var(--space-3xl);
  font-size: var(--font-size-sm);
  margin-top: var(--space-sm);
}

/* ============================================================
   登录
   ============================================================ */
.login-wrap {
  min-height: 100vh;
  display: flex;
  align-items: center;
  justify-content: center;
  background: var(--color-app-bg);
  background-image: var(--gradient-glow);
  padding: var(--space-4xl);
}
.login-card {
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  box-shadow: var(--shadow-card);
  padding: var(--space-6xl);
  width: var(--width-login);
  display: flex;
  flex-direction: column;
  gap: var(--space-xl);
}
.login-logo {
  display: flex;
  align-items: center;
  gap: var(--space-xl);
  justify-content: center;
  margin-bottom: var(--space-md);
}
.login-card h1 {
  font-size: var(--font-size-xl);
  font-weight: var(--font-weight-semibold);
  margin: 0;
  text-align: center;
}

/* 品牌 logo 方块 */
.brand-logo {
  width: var(--size-logo);
  height: var(--size-logo);
  flex: 0 0 auto;
  border-radius: var(--radius-md);
  background: var(--gradient-brand);
  color: var(--color-text-on-accent);
  display: inline-flex;
  align-items: center;
  justify-content: center;
  font-weight: var(--font-weight-bold);
  font-size: var(--font-size-sm);
  box-shadow: var(--shadow-glow);
}

/* ============================================================
   应用外壳：侧栏 + 内容
   ============================================================ */
.app {
  display: grid;
  grid-template-columns: var(--size-sidebar) 1fr;
  min-height: 100vh;
}

/* -------- 侧栏 -------- */
.sidebar {
  display: flex;
  flex-direction: column;
  background: var(--color-sidebar-bg);
  border-right: 1px solid var(--color-border);
  padding: var(--space-3xl) var(--space-xl);
  gap: var(--space-3xl);
}
.side-brand {
  display: flex;
  align-items: center;
  gap: var(--space-xl);
  padding: 0 var(--space-md);
}
.brand-name {
  font-weight: var(--font-weight-bold);
  font-size: var(--font-size-md);
  line-height: var(--line-height-tight);
}
.brand-sub {
  color: var(--color-text-muted);
  font-size: var(--font-size-xs);
}

.side-nav {
  display: flex;
  flex-direction: column;
  gap: var(--space-2xs);
  flex: 1;
}
/* 导航项（沿用 .tab data-tab 机制） */
.tab {
  display: flex;
  align-items: center;
  gap: var(--space-xl);
  width: 100%;
  text-align: left;
  background: transparent;
  color: var(--color-text-muted);
  border-radius: var(--radius-md);
  padding: var(--space-lg) var(--space-xl);
  font-size: var(--font-size-base);
  font-weight: var(--font-weight-medium);
}
.tab:hover {
  background: var(--color-white-a06);
  color: var(--color-text-primary);
}
.tab.active {
  background: var(--color-brand-soft);
  color: var(--color-text-primary);
  box-shadow: inset var(--indicator-width) 0 0 0 var(--color-brand);
}
.tab svg {
  width: var(--size-nav-icon);
  height: var(--size-nav-icon);
  flex: 0 0 auto;
}
.tab.active svg { color: var(--color-brand); }

/* 侧栏底部 */
.side-foot {
  display: flex;
  flex-direction: column;
  gap: var(--space-xl);
}
.side-status {
  display: flex;
  align-items: center;
  gap: var(--space-md);
  padding: var(--space-lg) var(--space-xl);
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-md);
  font-size: var(--font-size-xs);
  color: var(--color-text-muted);
}
.side-status b {
  color: var(--color-success);
  font-size: var(--font-size-base);
  font-weight: var(--font-weight-semibold);
}
.user-card {
  display: flex;
  align-items: center;
  gap: var(--space-lg);
  padding: var(--space-md);
  border-top: 1px solid var(--color-border);
}
.avatar {
  width: var(--size-avatar);
  height: var(--size-avatar);
  flex: 0 0 auto;
  border-radius: var(--radius-full);
  background: var(--gradient-brand);
  color: var(--color-text-on-accent);
  display: inline-flex;
  align-items: center;
  justify-content: center;
  font-size: var(--font-size-xs);
  font-weight: var(--font-weight-semibold);
}
.user-meta {
  display: flex;
  flex-direction: column;
  min-width: 0;
  flex: 1;
}
.me {
  font-size: var(--font-size-sm);
  font-weight: var(--font-weight-medium);
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
}
.me-role { font-size: var(--font-size-2xs); }
.icon-btn {
  background: transparent;
  border: 0;
  color: var(--color-text-muted);
  padding: var(--space-sm);
  min-height: var(--size-hit);
  min-width: var(--size-hit);
  border-radius: var(--radius-md);
  display: inline-flex;
  align-items: center;
  justify-content: center;
}
.icon-btn:hover { background: var(--color-white-a06); color: var(--color-text-primary); }
.icon-btn svg { width: var(--size-nav-icon); height: var(--size-nav-icon); }

.side-version {
  text-align: center;
  color: var(--color-text-muted);
  font-size: var(--font-size-2xs);
  padding-top: var(--space-md);
}

/* 测试数据开关：开启态高亮 */
#testdata-btn.on {
  background: var(--color-brand-soft);
  border-color: var(--color-brand);
  color: var(--color-brand);
}

/* -------- 内容区 -------- */
.content { display: flex; flex-direction: column; min-width: 0; }

.topbar {
  display: flex;
  align-items: center;
  gap: var(--space-2xl);
  min-height: var(--size-topbar);
  padding: var(--space-2xl) var(--space-4xl);
  border-bottom: 1px solid var(--color-border);
}
.page-head { display: flex; flex-direction: column; gap: var(--space-2xs); }
#page-title {
  margin: 0;
  font-size: var(--font-size-xl);
  font-weight: var(--font-weight-bold);
}
.page-sub { color: var(--color-text-muted); font-size: var(--font-size-sm); }
.topbar-right {
  margin-left: auto;
  display: flex;
  align-items: center;
  gap: var(--space-xl);
}
.stat-pill {
  display: flex;
  align-items: center;
  gap: var(--space-md);
  padding: var(--space-sm) var(--space-2xl);
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-full);
  font-size: var(--font-size-sm);
  color: var(--color-text-muted);
}
.stat-pill b { color: var(--color-success); font-weight: var(--font-weight-semibold); }

.panes { padding: var(--space-4xl); }
.pane { animation: fade-in var(--motion-slow) var(--motion-ease); }
.pane h3 {
  margin: var(--space-4xl) 0 var(--space-xl);
  font-size: var(--font-size-base);
  font-weight: var(--font-weight-semibold);
  color: var(--color-text-primary);
}
.pane h3:first-child { margin-top: 0; }

@keyframes fade-in {
  from { opacity: 0; transform: translateY(var(--space-md)); }
  to   { opacity: 1; transform: translateY(0); }
}

/* -------- 工具栏 -------- */
.toolbar {
  display: flex;
  align-items: center;
  gap: var(--space-2xl);
  margin-bottom: var(--space-2xl);
  flex-wrap: wrap;
}
.toolbar #q { width: var(--width-search); max-width: 100%; }
.toolbar label {
  display: flex;
  align-items: center;
  gap: var(--space-sm);
  color: var(--color-text-muted);
  font-size: var(--font-size-sm);
}

/* ============================================================
   表格（卡片式）
   ============================================================ */
.table-wrap {
  overflow-x: auto;
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  box-shadow: var(--shadow-card);
}
table {
  width: 100%;
  border-collapse: collapse;
  min-width: var(--table-min-width);
  font-size: var(--font-size-sm);
}
th, td {
  padding: var(--space-xl) var(--space-2xl);
  text-align: left;
  border-bottom: 1px solid var(--color-border);
  white-space: nowrap;
}
th {
  background: var(--color-surface-2);
  color: var(--color-text-muted);
  font-weight: var(--font-weight-semibold);
  font-size: var(--font-size-xs);
  text-transform: uppercase;
  letter-spacing: 0.02em;
}
tbody tr { transition: background var(--motion-fast) var(--motion-ease); }
tbody tr:last-child td { border-bottom: 0; }
tbody tr:hover { background: var(--color-white-a02); }
td.mono { font-family: var(--font-family-mono); font-size: var(--font-size-xs); }
.copy { cursor: pointer; color: var(--color-brand); }
.copy:hover { text-decoration: underline; }

/* 状态点 */
.dot {
  display: inline-block;
  width: var(--size-dot);
  height: var(--size-dot);
  border-radius: var(--radius-full);
  margin-right: var(--space-md);
  vertical-align: baseline;
}
.dot.on { background: var(--color-success); box-shadow: var(--shadow-glow); }
.dot.off { background: var(--color-danger); }
.dot.warn { background: var(--color-warning); }

/* 档位 / 状态 chip */
.badge {
  display: inline-block;
  padding: var(--space-2xs) var(--space-lg);
  border-radius: var(--radius-full);
  font-size: var(--font-size-xs);
  font-weight: var(--font-weight-medium);
  border: 1px solid transparent;
}
.badge.res { color: var(--color-info); background: var(--color-info-soft); }
.badge.g4  { color: var(--color-warning); background: var(--color-warning-soft); }
.badge.danger { color: var(--color-danger); background: var(--color-danger-soft); }
.badge.p2p { color: var(--color-success); background: var(--color-success-soft); }
.badge.relay { color: var(--color-warning); background: var(--color-warning-soft); }

/* 线路自检结果 */
.lt-verdict { font-size: var(--font-size-lg); font-weight: var(--font-weight-bold); margin-bottom: var(--space-sm); }
.lt-verdict.ok { color: var(--color-success); }
.lt-verdict.bad { color: var(--color-danger); }
.lt-reason { font-size: var(--font-size-sm); color: var(--color-text-muted); margin-bottom: var(--space-xl); }
.lt-table { width: 100%; border-collapse: collapse; font-size: var(--font-size-sm); }
.lt-table td { padding: var(--space-md) var(--space-sm); border-bottom: 1px solid var(--color-border); }
.lt-table tr:last-child td { border-bottom: 0; }
.lt-table td:first-child { width: 9ch; white-space: nowrap; }

.empty { padding: var(--space-6xl); text-align: center; color: var(--color-text-muted); }

/* IP来源就地编辑 */
.src-cell { cursor: text; border-radius: var(--radius-sm); }
.src-cell:hover { background: var(--color-white-a06); }
.src-in {
  width: 100%;
  padding: var(--space-xs) var(--space-sm);
  min-height: auto;
  font-size: var(--font-size-sm);
  border-color: var(--color-brand);
}

/* 操作列 & 小按钮 */
td.acts { white-space: nowrap; }
.mini {
  padding: var(--space-xs) var(--space-lg);
  min-height: auto;
  font-size: var(--font-size-xs);
  margin-right: var(--space-xs);
  border-radius: var(--radius-sm);
}

/* ============================================================
   弹窗
   ============================================================ */
.modal-wrap {
  position: fixed;
  inset: 0;
  background: var(--color-overlay-scrim);
  display: flex;
  align-items: center;
  justify-content: center;
  z-index: 50;
  padding: var(--space-4xl);
  animation: fade-in var(--motion-base) var(--motion-ease);
}
.modal {
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  box-shadow: var(--shadow-pop);
  padding: var(--space-4xl);
  width: var(--width-modal);
  max-width: 100%;
  display: flex;
  flex-direction: column;
  gap: var(--space-xl);
}
.modal h3 {
  margin: 0 0 var(--space-sm);
  font-size: var(--font-size-md);
  font-weight: var(--font-weight-semibold);
}
.modal label {
  display: flex;
  flex-direction: column;
  gap: var(--space-sm);
  font-size: var(--font-size-sm);
  color: var(--color-text-muted);
}
.modal label.ck { flex-direction: row; align-items: center; gap: var(--space-md); }
.modal input, .modal select { width: 100%; }
.modal-btns {
  display: flex;
  justify-content: flex-end;
  gap: var(--space-xl);
  margin-top: var(--space-md);
}

/* ============================================================
   Toast
   ============================================================ */
.toast {
  position: fixed;
  bottom: var(--space-5xl);
  left: 50%;
  transform: translateX(-50%);
  background: var(--color-surface);
  border: 1px solid var(--color-brand);
  color: var(--color-text-primary);
  padding: var(--space-xl) var(--space-4xl);
  border-radius: var(--radius-lg);
  box-shadow: var(--shadow-pop);
  z-index: 60;
  max-width: 80vw;
  font-size: var(--font-size-sm);
}

/* ============================================================
   首页：卡片 / KPI / 图表 / 中国地图
   ============================================================ */
.card-title {
  margin: 0 0 var(--space-xl);
  font-size: var(--font-size-md);
  font-weight: var(--font-weight-semibold);
  color: var(--color-text-primary);
}
.card-title.mt { margin-top: var(--space-4xl); }

/* KPI 指标卡 */
.kpi-row {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(11rem, 1fr));
  gap: var(--space-2xl);
  margin: var(--space-4xl) 0;
}
.kpi {
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  box-shadow: var(--shadow-card);
  padding: var(--space-3xl);
  border-left: var(--indicator-width) solid var(--color-brand);
}
.kpi-success { border-left-color: var(--color-success); }
.kpi-teal    { border-left-color: var(--color-teal); }
.kpi-info    { border-left-color: var(--color-info); }
.kpi-warning { border-left-color: var(--color-warning); }
.kpi-label { color: var(--color-text-muted); font-size: var(--font-size-sm); }
.kpi-value { font-size: var(--font-size-2xl); font-weight: var(--font-weight-bold); margin: var(--space-sm) 0; }
.kpi-sub { font-size: var(--font-size-xs); }

/* 图表网格 */
.chart-grid {
  display: grid;
  grid-template-columns: repeat(3, 1fr);
  gap: var(--space-2xl);
}
.chart-card {
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  box-shadow: var(--shadow-card);
  padding: var(--space-3xl);
  min-width: 0;
}
.chart-card.span2 { grid-column: span 2; }
.chart-card.span3 { grid-column: span 3; }
.chart-box { width: 100%; }
.chart-empty {
  display: flex;
  align-items: center;
  justify-content: center;
  min-height: 12ch;
  padding: var(--space-4xl);
  font-size: var(--font-size-sm);
}
.chart-svg { width: 100%; height: auto; display: block; }
.relay-chart { width: 100%; height: 15rem; }

/* 趋势折线 */
.chart-svg .grid { stroke: var(--color-border); stroke-width: 1; }
.chart-svg .ax { fill: var(--color-text-muted); font-size: var(--font-size-2xs); }
.chart-svg .area { fill: var(--color-brand); fill-opacity: 0.12; }
.chart-svg .line { fill: none; stroke: var(--color-brand); stroke-width: 2; stroke-linejoin: round; }
.chart-svg .pt { fill: var(--color-brand); stroke: var(--color-surface); stroke-width: 1.5; }

/* 环形图 */
.donut-wrap { display: flex; align-items: center; gap: var(--space-3xl); flex-wrap: wrap; }
.donut { width: 8.75rem; height: 8.75rem; flex: 0 0 auto; }
.dn-track { fill: none; stroke: var(--color-surface-3); stroke-width: 14; }
.dn-on { fill: none; stroke: var(--color-success); stroke-width: 14; stroke-linecap: round; }
.dn-num { fill: var(--color-text-primary); font-size: var(--font-size-2xl); font-weight: var(--font-weight-bold); }
.dn-cap { fill: var(--color-text-muted); font-size: var(--font-size-xs); }
.legend { display: flex; flex-direction: column; gap: var(--space-md); }
.legend .lg, .map-legend .lg { display: flex; align-items: center; gap: var(--space-md); font-size: var(--font-size-sm); color: var(--color-text-muted); }
.legend b { color: var(--color-text-primary); }
.sw { width: var(--space-xl); height: var(--space-xl); border-radius: var(--radius-sm); flex: 0 0 auto; display: inline-block; }
.sw-on  { background: var(--color-success); }
.sw-off { background: var(--color-danger); }
.sw-res { background: var(--color-info); }
.sw-g4  { background: var(--color-warning); }

/* Top 省份条形 */
.bar-row { display: flex; align-items: center; gap: var(--space-xl); margin-bottom: var(--space-xl); font-size: var(--font-size-sm); }
.bar-name { width: 5ch; flex: 0 0 auto; color: var(--color-text-primary); }
.bar-track { flex: 1; height: var(--space-lg); background: var(--color-surface-3); border-radius: var(--radius-full); overflow: hidden; }
.bar-fill { display: block; height: 100%; background: var(--gradient-brand); border-radius: var(--radius-full); transition: width var(--motion-slow) var(--motion-ease); }
.bar-val { flex: 0 0 auto; color: var(--color-text-muted); font-size: var(--font-size-xs); min-width: 12ch; text-align: right; }

/* ---------- 中国地图 ---------- */
.map-card {
  background: var(--color-surface);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-xl);
  box-shadow: var(--shadow-card);
  padding: var(--space-3xl);
}
.map-head { display: flex; align-items: flex-start; justify-content: space-between; gap: var(--space-xl); margin-bottom: var(--space-xl); }
.map-body { position: relative; }
/* ECharts 画布容器（canvas 渲染，高度必须显式给） */
.map-svg { width: 100%; height: min(62vh, 42rem); }
/* 下钻侧栏面板 */
.map-panel {
  position: absolute;
  top: var(--space-md);
  right: var(--space-md);
  width: var(--width-search);
  max-width: 60%;
  max-height: 90%;
  overflow: auto;
  background: var(--color-surface-2);
  border: 1px solid var(--color-border);
  border-radius: var(--radius-lg);
  box-shadow: var(--shadow-pop);
  padding: var(--space-2xl);
}
.mp-title { font-size: var(--font-size-lg); font-weight: var(--font-weight-semibold); }
.mp-sub { font-size: var(--font-size-xs); margin: var(--space-sm) 0 var(--space-xl); }
.mp-list { display: flex; flex-direction: column; gap: var(--space-md); }
.mp-row { display: flex; justify-content: space-between; gap: var(--space-md); font-size: var(--font-size-sm); padding-bottom: var(--space-md); border-bottom: 1px solid var(--color-border); }
.mp-row:last-child { border-bottom: 0; }
.map-legend { display: flex; gap: var(--space-4xl); flex-wrap: wrap; margin-top: var(--space-xl); }
.mp-grad { width: var(--space-6xl); height: var(--space-xl); border-radius: var(--radius-sm); background: var(--gradient-brand); flex: 0 0 auto; display: inline-block; }

/* ============================================================
   响应式：窄屏折叠侧栏为顶部横排
   ============================================================ */
@media (max-width: 53.75em) {
  .app { grid-template-columns: 1fr; }
  .sidebar {
    flex-direction: row;
    flex-wrap: wrap;
    align-items: center;
    gap: var(--space-xl);
    border-right: 0;
    border-bottom: 1px solid var(--color-border);
  }
  .side-nav { flex-direction: row; flex-wrap: wrap; flex: 1 1 100%; }
  .tab.active { box-shadow: inset 0 calc(-1 * var(--focus-width)) 0 0 var(--color-brand); }
  .side-foot { flex-direction: row; align-items: center; }
  .topbar { flex-wrap: wrap; }
  .chart-grid { grid-template-columns: 1fr; }
  .chart-card.span2, .chart-card.span3 { grid-column: span 1; }
  .map-panel { position: static; width: auto; max-width: none; margin-top: var(--space-xl); }
  .map-svg { height: 60vh; }
}

/* ============================================================
   可访问性：尊重减少动效偏好
   ============================================================ */
@media (prefers-reduced-motion: reduce) {
  *, *::before, *::after {
    animation-duration: 0.001ms !important;
    animation-iteration-count: 1 !important;
    transition-duration: 0.001ms !important;
  }
}
