/* =============================================================================
   Pisonics Cloud — UI 设计系统（后台 /admin + 门户 /portal 共用）
   方案 A：浅暖白底 + 柔和阴影 + 品牌绿主色(#16794f) + 品牌橙点缀(#f35425)
   两端共用同一套 token 与组件类，仅导航形式不同（后台侧栏 / 门户顶栏）。
   —— 由 Claude Design 交付稿（README）落地，像素级数值见 README「Design Tokens」。
   ========================================================================== */

/* 中文正文字体：Noto Sans SC（设计指定），经 Google Fonts 引入；display=swap 先用系统字体兜底再替换。
   注：@import 必须置于所有样式规则之前。若 mainland China 访问 fonts.googleapis.com 受限，
   会优雅回退到 --sans 里的 PingFang/系统字体；如需国内稳定加载，建议自托管（见文件末注）。 */
@import url('https://fonts.googleapis.com/css2?family=Noto+Sans+SC:wght@400;500;600;700&display=swap');

/* JetBrains Mono：数字/编号/时间戳/令牌/CRC/流量统一等宽（仓库已自带 woff2）。 */
@font-face{font-family:'JetBrains Mono';font-style:normal;font-weight:400;font-display:swap;
  src:url('../fonts/jetbrains-mono/jetbrains-mono-latin-400.woff2') format('woff2');}
@font-face{font-family:'JetBrains Mono';font-style:normal;font-weight:500;font-display:swap;
  src:url('../fonts/jetbrains-mono/jetbrains-mono-latin-500.woff2') format('woff2');}

:root{
  --bg:#f7f8f6; --surface:#fff;
  --card-bd:#eef0eb; --line:#f2f3f0; --line2:#ececea; --line3:#e9ebe6;
  --tx:#1f2a24; --tx2:#2a3630; --tx-sec:#6f7a73;
  --tx-weak:#8a948e; --tx-weak2:#9aa39d; --tx-weak3:#aab2ac;
  --green:#16794f; --green-soft:#eaf3ee; --green-face:#f3f8f5; --green-bd:#d4e7dd;
  --orange:#f35425;
  --blue:#2f6fd0; --blue-soft:#eaf1fb;
  --red:#c0392f; --red-soft:#fbeae8; --red-bd:#f2dad6;
  --amber-tx:#c4621a; --amber-soft:#fbf0e4;
  --notice-bg:#fdf6ec; --notice-bd:#f3e2c4; --notice-tx:#7a5a1e;
  --fld-bd:#e2e5e0;
  --sans:'Noto Sans SC',-apple-system,BlinkMacSystemFont,'Segoe UI','PingFang SC','Microsoft YaHei',sans-serif;
  --mono:'JetBrains Mono',ui-monospace,'SF Mono',Menlo,Consolas,monospace;
  --shadow-card:0 1px 2px rgba(20,30,20,.03);
  --shadow-seg:0 1px 2px rgba(0,0,0,.05);
}

/* ---- base ---- */
.pa-app,.pa-app *{box-sizing:border-box}
.pa-app{margin:0;font-family:var(--sans);color:var(--tx);background:var(--bg);
  display:flex;height:100vh;overflow:hidden;-webkit-font-smoothing:antialiased}
.pa-app a{text-decoration:none}
/* 链接默认色复位：用 :where() 让特异性≈0，既压过浏览器默认蓝链，又不会盖住
   .btn-pri / .pa-nav / .act 等组件自带的颜色（否则白字按钮会继承成深色）。 */
:where(.pa-app) a{color:inherit}
.mono{font-family:var(--mono);font-feature-settings:"tnum" 1}

/* =============================== 侧栏 Sidebar =============================== */
.pa-side{flex:0 0 248px;width:248px;background:var(--surface);border-right:1px solid var(--line3);
  padding:18px 16px;display:flex;flex-direction:column;
  transition:width .18s ease,padding .18s ease}
.brand-row{display:flex;align-items:center;gap:10px;height:34px;padding:0 4px;margin-bottom:6px}
.brand-word{height:21px;width:auto;display:block}
.brand-mark{display:none;width:34px;height:34px;border-radius:9px;background:var(--green-soft);
  align-items:center;justify-content:center;cursor:pointer;flex:0 0 34px}
.brand-mark img{width:21px;height:21px;display:block}
.collapse-btn{margin-left:auto;width:34px;height:34px;border:0;background:transparent;color:var(--tx-weak2);
  border-radius:8px;display:inline-flex;align-items:center;justify-content:center;cursor:pointer}
.collapse-btn:hover{background:#f1f3f0;color:var(--tx-sec)}

.nav-group{margin:22px 0 8px;padding:0 12px;font-size:11px;letter-spacing:.14em;font-weight:600;
  color:var(--tx-weak3);text-transform:uppercase}
.nav-group:first-of-type{margin-top:18px}
.pa-nav{display:flex;align-items:center;gap:12px;height:42px;padding:0 14px;border-radius:9px;
  font-size:14.5px;font-weight:500;color:#5b6660;margin-bottom:2px}
.pa-nav svg{width:18px;height:18px;flex:0 0 18px;stroke:currentColor;stroke-width:1.8;fill:none}
.pa-nav:hover{background:#f1f3f0;color:var(--tx2)}
.pa-nav[data-active="true"]{background:var(--green-soft);color:var(--green);font-weight:600}
.side-foot{margin-top:auto;padding-top:14px;border-top:1px solid #eef0eb;
  font-size:12.5px;color:var(--tx-weak);display:flex;align-items:center;gap:6px}
.side-foot a{display:inline-flex;align-items:center;gap:6px}
.side-foot a:hover{color:var(--green)}

/* 折叠态 */
.pa-side[data-collapsed="true"]{width:74px!important;flex-basis:74px!important;max-width:74px!important;
  min-width:0!important;overflow:hidden;padding-left:12px;padding-right:12px}
.pa-side[data-collapsed="true"] .nav-label,
.pa-side[data-collapsed="true"] .nav-group,
.pa-side[data-collapsed="true"] .brand-word,
.pa-side[data-collapsed="true"] .collapse-btn,
.pa-side[data-collapsed="true"] .side-foot .lbl{display:none!important}
.pa-side[data-collapsed="true"] .brand-mark{display:inline-flex}
.pa-side[data-collapsed="true"] .pa-nav{justify-content:center;padding:0;margin-bottom:4px}
.pa-side[data-collapsed="true"] .brand-row{justify-content:center;margin-bottom:18px;padding:0}
.pa-side[data-collapsed="true"] .side-foot{justify-content:center}

/* =============================== 主区 =============================== */
.pa-main{flex:1;display:flex;flex-direction:column;min-width:0}
.pa-top{height:62px;flex:0 0 62px;background:var(--surface);border-bottom:1px solid var(--line2);
  padding:0 26px;display:flex;align-items:center;gap:14px}
.pa-top .title{font-size:16px;font-weight:600;color:#23211c}
.pa-top .spacer{flex:1}
.icon-btn{position:relative;width:34px;height:34px;border:1px solid #e6e8e3;background:var(--surface);
  border-radius:8px;color:var(--tx-sec);display:inline-flex;align-items:center;justify-content:center;cursor:pointer}
.icon-btn:hover{background:#f4f6f3}
.icon-btn .dot{position:absolute;top:-3px;right:-3px;width:8px;height:8px;border-radius:50%;
  background:var(--orange);border:2px solid #fff}
.top-sep{width:1px;height:24px;background:#e8eae5}
.avatar{width:32px;height:32px;border-radius:50%;background:var(--green-soft);color:var(--green);
  display:inline-flex;align-items:center;justify-content:center;font-weight:600;font-size:14px}
.user-meta{line-height:1.25}.user-meta .nm{font-size:13.5px;color:var(--tx2);font-weight:500}
.user-meta .rl{font-size:11px;color:var(--tx-weak2)}
.lnk{font-size:13.5px;color:var(--tx-sec)}.lnk:hover{color:var(--orange)}

.pa-content{flex:1;overflow:auto;padding:26px 30px}
.pa-content.narrow{max-width:1040px}
.pa-content.narrow2{max-width:920px}

/* 页头 */
.page-head{display:flex;align-items:flex-start;justify-content:space-between;gap:16px;margin-bottom:22px}
.page-head h2{margin:0;font-size:24px;font-weight:700;color:var(--tx);line-height:1.2}
.page-head .sub{margin:6px 0 0;font-size:14px;color:var(--tx-weak)}
.page-head .sub .mono{font-size:13px}
.head-actions{display:flex;align-items:center;gap:10px;flex:0 0 auto}
.crumb{font-size:13px;color:var(--tx-weak);margin-bottom:10px}
.crumb b{color:var(--tx2);font-weight:600}

/* =============================== 组件 =============================== */
.card{background:var(--surface);border:1px solid var(--card-bd);border-radius:14px;box-shadow:var(--shadow-card)}
.card-pad{padding:20px 22px}
.card-h{display:flex;align-items:center;justify-content:space-between;gap:12px;margin-bottom:14px}
.card-h h3{margin:0;font-size:15px;font-weight:600;color:var(--tx)}
.card-h .sub{font-size:12.5px;color:var(--tx-weak2)}

/* 按钮 */
.btn-pri,.btn-gh{display:inline-flex;align-items:center;gap:7px;height:38px;padding:0 16px;border-radius:9px;
  font-size:13.5px;cursor:pointer;border:1px solid transparent;white-space:nowrap;transition:all .14s}
.btn-pri{background:var(--green);color:#fff;font-weight:600}
.btn-pri:hover{background:#125f3e}
.btn-gh{background:var(--surface);color:#46524b;border-color:var(--fld-bd);font-weight:500}
.btn-gh:hover{background:#f4f6f3;border-color:#cfd4cd}
.btn-pri svg,.btn-gh svg{width:15px;height:15px;stroke:currentColor;stroke-width:2;fill:none}
.btn-danger-out{display:inline-flex;align-items:center;gap:7px;height:38px;padding:0 16px;border-radius:9px;
  font-size:13.5px;font-weight:500;cursor:pointer;background:#fff;color:var(--red);border:1px solid var(--red-bd)}
.btn-danger-out:hover{background:#fdf3f1}

/* 表单 */
.fld{width:100%;height:42px;border:1px solid var(--fld-bd);border-radius:9px;padding:0 14px;font-size:14px;
  background:#fff;color:var(--tx);transition:border-color .15s,box-shadow .15s;font-family:inherit}
textarea.fld{height:auto;min-height:84px;padding:10px 14px;line-height:1.6}
select.fld{appearance:none;background-image:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 24 24' fill='none' stroke='%239aa39d' stroke-width='2'%3E%3Cpath d='M6 9l6 6 6-6'/%3E%3C/svg%3E");background-repeat:no-repeat;background-position:right 14px center;padding-right:34px}
.fld.mono{font-family:var(--mono)}
.fld:focus{outline:none;border-color:var(--green);box-shadow:0 0 0 3px rgba(22,121,79,.1)}
.flbl{display:block;font-size:13px;color:var(--tx-sec);margin-bottom:7px}
.form-grid{display:grid;grid-template-columns:1fr 1fr;gap:18px 24px}
.form-grid .span2{grid-column:1 / -1}
.form-actions{display:flex;align-items:center;gap:10px;margin-top:22px;padding-top:18px;border-top:1px solid var(--line)}
.form-actions .spacer{flex:1}
.field-hint{font-size:12px;color:var(--tx-weak2);margin-top:6px}

/* 开关 toggle（隐藏 checkbox + label 渲染胶囊） */
.tgl{position:relative;display:inline-block;width:38px;height:22px;flex:0 0 38px}
.tgl input{position:absolute;opacity:0;width:0;height:0}
.tgl .track{position:absolute;inset:0;background:#cfd6d0;border-radius:20px;transition:background .15s}
.tgl .track::after{content:"";position:absolute;top:2px;left:2px;width:18px;height:18px;border-radius:50%;
  background:#fff;box-shadow:0 1px 2px rgba(0,0,0,.2);transition:transform .15s}
.tgl input:checked + .track{background:var(--green)}
.tgl input:checked + .track::after{transform:translateX(16px)}
.tgl-row{display:flex;align-items:center;gap:12px}

/* 分段控件 seg */
.seg{display:inline-flex;background:#f3f5f2;padding:3px;border-radius:9px;gap:2px}
.seg button,.seg a{border:0;background:transparent;font-size:12.5px;padding:5px 11px;border-radius:6px;
  color:var(--tx-sec);cursor:pointer;font-family:inherit;line-height:1}
.seg .seg-on,.seg button.seg-on{background:#fff;color:var(--green);font-weight:600;box-shadow:var(--shadow-seg)}

/* 搜索框 */
.search{position:relative;display:inline-flex;align-items:center}
.search svg{position:absolute;left:12px;width:16px;height:16px;stroke:var(--tx-weak2);stroke-width:2;fill:none}
.search input{height:38px;width:240px;border:1px solid var(--fld-bd);border-radius:9px;padding:0 12px 0 34px;font-size:13.5px;background:#fff}
.search input:focus{outline:none;border-color:var(--green);box-shadow:0 0 0 3px rgba(22,121,79,.1)}

/* 表格 */
.tbl{width:100%;border-collapse:collapse;font-size:13.5px}
.tbl th{font-size:12px;font-weight:500;color:var(--tx-weak2);text-align:left;padding:12px 14px;white-space:nowrap}
.tbl td{padding:13px 14px;border-top:1px solid var(--line);color:var(--tx2);vertical-align:middle}
.tbl tbody tr:hover{background:#fafbfa}
.tbl .r{text-align:right}.tbl .c{text-align:center}
.tbl .empty{text-align:center;color:var(--tx-weak2);padding:40px 0}
.card .tbl th:first-child,.card .tbl td:first-child{padding-left:22px}
.card .tbl th:last-child,.card .tbl td:last-child{padding-right:22px}

/* 行内操作 */
.acts{display:inline-flex;gap:8px;justify-content:flex-end}
.act{display:inline-flex;align-items:center;gap:5px;height:30px;padding:0 11px;border-radius:8px;
  border:1px solid #e6e8e3;background:#fff;color:#5b6660;font-size:12.5px;font-weight:500;cursor:pointer;transition:all .13s}
.act svg{width:13px;height:13px;stroke:currentColor;stroke-width:1.9;fill:none}
.act:hover{border-color:#bcd8ca;color:var(--green);background:var(--green-face)}
.act-pri{color:var(--green);border-color:var(--green-bd);background:var(--green-face)}
.act-pri:hover{border-color:var(--green);background:var(--green-soft)}
.act-danger{color:#a4554b}
.act-danger:hover{border-color:var(--red-bd);color:var(--red);background:#fdf3f1}

/* 徽章 / 状态点 */
.badge{display:inline-flex;align-items:center;font-size:11.5px;padding:3px 8px;border-radius:5px;font-weight:500;white-space:nowrap}
.bg-green{background:var(--green-soft);color:var(--green)}
.bg-blue{background:var(--blue-soft);color:var(--blue)}
.bg-red{background:var(--red-soft);color:var(--red)}
.bg-amber{background:var(--amber-soft);color:var(--amber-tx)}
.bg-grey{background:#f0f2ef;color:#5b6660}
.role-badge{font-family:var(--mono);letter-spacing:.04em;background:var(--green-soft);color:var(--green);
  font-size:11.5px;padding:3px 8px;border-radius:5px;font-weight:500}
.dot{display:inline-block;width:7px;height:7px;border-radius:50%;vertical-align:middle}
.dot-green{background:var(--green)}.dot-grey{background:#c4ccc6}.dot-orange{background:var(--orange)}.dot-blue{background:var(--blue)}
.on-pill{display:inline-flex;align-items:center;gap:6px;font-size:12.5px;color:var(--green)}
.off-txt{font-size:12.5px;color:var(--tx-weak2)}
.status-pill{display:inline-flex;align-items:center;gap:6px;padding:4px 11px;border-radius:20px;font-size:12.5px;font-weight:500}
.status-pill.on{background:var(--green-soft);color:var(--green)}
.status-pill.off{background:#f0f2ef;color:var(--tx-sec)}

/* KPI */
.kpi-row{display:grid;grid-template-columns:repeat(4,1fr);gap:16px;margin-bottom:16px}
.kpi{padding:18px 20px}
.kpi .k-lbl{font-size:13px;color:var(--tx-sec)}
.kpi .k-val{font-size:34px;font-weight:600;color:var(--tx);font-family:var(--mono);line-height:1.1;margin-top:8px}
.kpi .k-val .u{font-size:14px;color:var(--tx-weak2);font-family:var(--sans);font-weight:500;margin-left:4px}
.kpi .k-sub{font-size:12.5px;color:var(--tx-weak2);margin-top:6px}
.kpi .k-sub .warn{color:var(--orange);font-weight:500}
.kpi-bar{height:5px;border-radius:3px;background:#eef2ef;margin-top:12px;overflow:hidden}
.kpi-bar i{display:block;height:100%;background:var(--green);border-radius:3px}
.readout .k-val{font-size:30px}

/* notice / alert 条 */
.notice{background:var(--notice-bg);border:1px solid var(--notice-bd);color:var(--notice-tx);
  border-radius:12px;padding:14px 16px;line-height:1.7;font-size:13.5px}
.flash-ok{background:var(--green-soft);border:1px solid var(--green-bd);color:#2a5a44;
  border-radius:10px;padding:11px 14px;font-size:13.5px;margin-bottom:16px}
.info-strip{background:var(--green-soft);border:1px solid var(--green-bd);color:#2a5a44;
  border-radius:10px;padding:11px 14px;font-size:13px}

/* 进度条（OTA） */
.prog{height:6px;border-radius:6px;background:#eef2ef;overflow:hidden}
.prog i{display:block;height:100%;background:var(--green);border-radius:6px}
.prog.fail{background:var(--red-soft)}.prog.fail i{background:var(--red)}
.prog-sub{font-size:11.5px;color:var(--tx-weak2);margin-top:5px;font-family:var(--mono)}

/* 图表 */
.chart{width:100%}
.audit-line{display:flex;gap:11px;padding:10px 0;border-top:1px solid var(--line)}
.audit-line:first-child{border-top:0}
.audit-line .d{margin-top:6px;flex:0 0 7px}
.audit-line .m{font-size:13px;color:var(--tx2)}
.audit-line .s{font-size:12px;color:var(--tx-weak2);margin-top:2px}
.audit-line .t{margin-left:auto;font-size:11.5px;color:var(--tx-weak2);font-family:var(--mono);white-space:nowrap}

.grid-16{display:grid;gap:16px}
.grid-chart{grid-template-columns:1.6fr 1fr}
@media(max-width:1000px){.grid-chart{grid-template-columns:1fr}.kpi-row{grid-template-columns:repeat(2,1fr)}}

/* =============================== 门户 /portal =============================== */
.pt-app{margin:0;font-family:var(--sans);color:var(--tx);background:var(--bg);min-height:100vh;
  -webkit-font-smoothing:antialiased}
.pt-app *{box-sizing:border-box}.pt-app a{text-decoration:none}
:where(.pt-app) a{color:inherit}
.pt-top{height:64px;background:var(--surface);border-bottom:1px solid var(--line2);
  padding:0 32px;display:flex;align-items:center;gap:24px}
.pt-top .brand{display:flex;align-items:center;gap:10px}
.pt-top .brand img{width:26px;height:26px}
.pt-top .brand b{font-size:17px;font-weight:600;color:var(--tx)}
.pt-top .spacer{flex:1}
.topnav{font-size:14.5px;color:#5b6660}.topnav:hover,.topnav[data-active="true"]{color:var(--green)}
.pt-content{margin:0;padding:32px 36px}
.pt-scroll{flex:1;min-height:0;overflow-y:auto}
.pt-content h2{font-size:27px;font-weight:700;margin:0 0 6px}
.pt-content .sub{font-size:14px;color:var(--tx-weak);margin:0 0 22px}
.back-lnk{display:inline-flex;align-items:center;gap:6px;font-size:13.5px;color:var(--tx-sec);margin-bottom:14px}
.back-lnk:hover{color:var(--green)}

/* 门户顶栏：语言 + 用户下拉菜单 */
.pt-dd{position:relative}
.pt-dd-btn{display:inline-flex;align-items:center;gap:7px;height:36px;padding:0 11px;border:1px solid var(--line2);background:var(--surface);border-radius:9px;color:var(--tx-sec);font-size:14px;font-family:inherit;line-height:1;cursor:pointer}
.pt-dd-btn:hover,.pt-dd[data-open="true"] .pt-dd-btn{background:var(--green-face);border-color:var(--green-bd)}
.pt-dd-btn .cv{color:var(--tx-weak2);transition:transform .15s}
.pt-dd[data-open="true"] .pt-dd-btn .cv{transform:rotate(180deg)}
.pt-avatar{width:26px;height:26px;border-radius:50%;background:var(--green-soft);color:var(--green);display:inline-flex;align-items:center;justify-content:center;font-weight:600;font-size:12.5px;flex:0 0 auto}
.pt-dd-menu{position:absolute;top:calc(100% + 8px);right:0;min-width:204px;background:var(--surface);border:1px solid var(--line2);border-radius:12px;box-shadow:0 10px 30px rgba(20,40,30,.12);padding:6px;display:none;z-index:60}
.pt-dd[data-open="true"] .pt-dd-menu{display:block}
.pt-dd-menu a{display:flex;align-items:center;gap:9px;padding:9px 11px;border-radius:8px;font-size:14px;color:var(--tx2);white-space:nowrap}
.pt-dd-menu a:hover{background:var(--green-face);color:var(--green)}
.pt-dd-menu a[data-active="true"]{color:var(--green);font-weight:600;background:var(--green-soft)}
.pt-dd-menu .tick{margin-left:auto;color:var(--green)}
.pt-dd-head{padding:9px 11px 9px;margin-bottom:4px;border-bottom:1px solid var(--line)}
.pt-dd-head .nm{font-size:14px;font-weight:600;color:var(--tx)}
.pt-dd-head .rl{font-size:12px;color:var(--tx-weak2);margin-top:2px}
.pt-dd-sep{height:1px;background:var(--line);margin:4px 0}

/* 门户表单略大 */
.fld-lg{height:46px;border-radius:10px}

/* 居中登录 */
.center-wrap{min-height:100vh;display:flex;flex-direction:column;align-items:center;justify-content:center;padding:24px}

/* ── 自托管 Noto Sans SC（国内稳定加载，可选）──────────────────────────────
   1) 用 google-webfonts-helper 或 fontsource 下载 noto-sans-sc 的 woff2(含 unicode-range
      子集)，放到 public/fonts/noto-sans-sc/（参照仓库已有的 noto-serif-sc 目录结构）。
   2) 删除文件顶部的 @import，改为本地 @font-face（weight 400/500/600/700）。
   3) --sans 已把 'Noto Sans SC' 列在首位，无需改动。
   ──────────────────────────────────────────────────────────────────────── */
