// ==UserScript== // @name TP路由器增强 // @name:en Bro-Stat-TP // @namespace ucxn // @version 5.9.2 // @description 哥哥科技 QQ群 680464365 // @description:en https://github.com/ucxn/Mi-Stat_Max // @author 哥哥科技 space.bilibili.com/501430041 // @noframes // @icon https://scriptcat.org/api/v2/resource/image/duygQktL5QjWtkLc // @include http://10.*.*.* // @include http://192.168.*.* // @include /^https?:\/\/tplogin\.[a-z]+\/.*$/ // @include http://172.16.* // @include https://10.*.*.* // @include https://192.168.*.* // @match *://tplogin.cn/* // @include https://172.16.* // @run-at document-start // @grant GM_setValue // @grant GM_getValue // @storageName GBNPA_Storage // @license AGPL-3.0-or-later // ==/UserScript== (function () { 'use strict'; console.log("🚀 哥哥科技 V5.9.9 终极引擎已装载..."); function escapeHTML(str) { if (!str) return ''; return String(str).replace(/[&<>'"]/g, function (match) { return { '&': '&', '<': '<', '>': '>', "'": ''', '"': '"' } [match]; }); } // ======== [0] 用户极客环境变量配置区 ======== const CONFIG = { forceMeshMode: 0, // 【Mesh探测模式】0: 官方拓扑驱动 | 1: n秒智能等待(默认) | 2: 强制大包抓取(专治阉割、不出数据) uiLayout: 2,//【面板拓扑结构】 0: 经典版 | 1: 详细紧凑版(驾驶舱美学) | 2: 详细平铺版(报表流美学) injectMode: 3, //1: 优先,10秒悬浮舱(D)| 3:强制模式 calcMode: 1,// 1: 上行/下行倍数模式, 0: 上行占总和比例模式 ratioExtremeUp: 10, // 极端上传判定阈值 (> 1000%) ratioWarnUp: 0.07, // 重度上传警告阈值 (> 7%) ratioExtremeDown: 0.01, // 极端下载判定阈值 (< 1%) ratioThreshold: 7, // (仅calcMode=0时有效) 上传占比报警阈值(%) readSaveData: 1, // 【开关切换】 1: 读档模式(继承本次历史量) | 0: 新局模式(从打开网页此刻归零重新计流) lanRefreshInterval: 3, // LAN口刷新时间(秒),用于精准补偿0到唤醒时的瞬时流量 wanRefreshInterval: 3, // 【新增】WAN口刷新时间(秒),用于精准补偿0到唤醒时的瞬时流量 portMap: { "eth1": "网口 1", "eth2": "网口 2", "eth3": "网口 3", "eth4": "网口 4", "wl0": "2.4G", "wl1": "5.2G", "wl2": "5.8G" } }; const __origOpen = XMLHttpRequest.prototype.open; XMLHttpRequest.prototype.open = function(method, url) { let m = String(url).match(/stok=([a-zA-Z0-9]+)/i); if (m) window.__tpStok = m[1]; return __origOpen.apply(this, arguments); }; let _saved = null; if (CONFIG.readSaveData === 1 && typeof GM_getValue !== 'undefined') { try { _saved = GM_getValue('ha_snapshot', null); } catch(e) {console.warn(e)} } const S = { lt: 0, wInstUp: 0, wInstDn: 0, // [修改] 启动时将 WAN口 总量继承为存档数据 wTotUp: _saved?.global?.wan_up || 0, wTotDn: _saved?.global?.wan_down || 0, cls: {}, isPinned: !0, w2U: 0, w2D: 0, w2TotUp: 0, w2TotDn: 0, w2LT: undefined, hasW2: !1, is5G_149: !1 }; const _w = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; if (location.pathname.includes('main.html')) { const jumpToPc = (stok) => { if (!window.__gegeJumped && location.hash.includes('#/home')) { window.__gegeJumped = !0; document.body.innerHTML = '
🚀 哥哥科技:正在强制跃迁至 PC 网页版...
'; location.replace(`/cgi-bin/luci/;stok=${stok}/web?goto=pc#router`); } }; const origOpen = _w.XMLHttpRequest.prototype.open; _w.XMLHttpRequest.prototype.open = function(method, url) { let m = String(url).match(/;stok=([a-fA-F0-9]+)/); if (m) jumpToPc(m[1]); return origOpen.apply(this, arguments); }; const origFetch = _w.fetch; if (origFetch) { _w.fetch = function() { let m = String(arguments[0]).match(/;stok=([a-fA-F0-9]+)/); if (m) jumpToPc(m[1]); return origFetch.apply(this, arguments); }; } } function fB(bps) { if (bps > 1e9) return `${(bps * 1e-6).toFixed(1)} Mbit/s`; if (bps > 1e6) return `${(bps * 1e-6).toFixed(2)} Mbps`; if (bps > 1e3) return `${(bps * 1e-3).toFixed(1)} Kbps`; return `${Math.round(bps)} bps`; } function fBy(bps) { if (bps === 0) return '0 B'; if (bps > 8388608) return `${(bps / 8388608).toFixed(2)} MiB/s`; return bps < 8602 ? ((bps * 0.001 | 0) === bps * 0.001 ? `${['0', '[1/8]', '[2/8]', '[3/8]', '[4/8]', '[5/8]', '[6/8]', '[7/8]', '[1]'][bps * 0.001]} KB/s` : `${(bps * 0.000125).toFixed(2)} KB/s`) : `${(bps / 8192).toFixed(1)} KB/s`; } function fV(bits) { if (bits > 83886080000) return `${(bits / 8589934592).toFixed(4)} GB`; if (bits > 8388608000) return `${(bits / 8388608).toFixed(1)} MB`; if (bits > 8388608) return `${(bits / 8388608).toFixed(4)} MB`; if (bits > 8192) return `${(bits / 8192).toFixed(3)} KB`; return `${Math.round(bits / 8)} B`; } function fSV(bits) { if (bits >= 84607500288) return `${(bits / 8589934592).toPrecision(4)}GB`; if (bits > 8388608000) return `${Math.round(bits / 8388608)}MB`; if (bits > 8388608) return `${(bits / 8388608).toFixed(2)}MB`; if (bits >= 8192) return `${(bits / 8192).toFixed(1)}KB`; return `${Math.round(bits / 8)}B`;} function fOT(totalSec) { totalSec = Math.floor(totalSec); if (totalSec < 0) return ""; const d = Math.floor(totalSec / 86400); let r = totalSec - d * 86400; const h = Math.floor(r / 3600); r = r - h * 3600; const m = Math.floor(r / 60); const s = r - m * 60; return d > 0 ? `${d}天${h}时${m}分${s}秒` : `${h}小时${m}分${s}秒`;} function nM(m) { return m ? m.toLowerCase().replace(/-/g, ':').replace(/\s/g, '') : ''; } const st = document.createElement('style'); st.innerHTML = `.config-item{ clear:both;}.config-item-box{display:flex!important; align-items:stretch!important;padding-bottom: 12px!important;}.config-item .logo{width:33%!important; float:none!important;display:flex!important;flex-direction:row;}.config-item .dev-intro{flex:1;display:flex!important;flex-direction:column;justify-content:flex-start;min-height:50px;padding-bottom:0!important;margin-bottom:0!important;}.config-item .info{width:27%!important;float:none!important;display:flex!important;flex-direction:column;justify-content:flex-start;padding:0 10px!important;border-right:1px solid #eee;}.config-item .speed{width:40%!important;float:none!important;display:flex!important;flex-direction:column;justify-content:center;padding:0 10px!important;}.geek-row{display:flex;justify-content:space-between;align-items:center;white-space:nowrap;height:20px;} .geek-label{width:110px;color:#333;font-weight:bold;}.geek-val-box{flex:1;display:flex;gap:15px;margin-left:10px;}.geek-fixed-width{display:inline-block;width:120px;}.geek-right-box{text-align:right;min-width:220px;font-weight:bold;}.c-up{color:#ff4c00;}.c-down{color:#0059fa;}.gege-up-box,.gege-down-box{margin-top:auto!important;margin-bottom:0!important;width:95%;}.gege-ratio-box{margin-top:10px;width:95%;margin-bottom:5px;}.t-row{font-size:12px;font-weight:bold;margin-bottom:2px;display:flex;justify-content:space-between;font-family:system-ui, sans-serif;}.zte-thin-bar{width:100%;height:3px;background:rgba(0,0,0,0.05);border-radius:1.5px;overflow:hidden;}.zte-thin-bar-inner{height:100%;transition:width 0.5s ease-out;}.zte-thin-bar-inner.up{background:#ff4c00;}.zte-thin-bar-inner.down{background:#0059fa;}.gege-ratio-top{display:flex;justify-content:space-between;font-size:12px;font-weight:bold;margin-bottom:2px;}.gege-ratio-bar{width:100%;height:4px;background:#0059fa;border-radius:2px;overflow:hidden;}.gege-ratio-bar-inner{height:100%;background:#ff4c00;transition:width 0.5s ease-out;}.zte-enhance-speed{display:flex;flex-direction:column;gap:6px;width:100%;font-family:system-ui, sans-serif;} .zte-bar-wrap{position:relative;width:100%;border-radius:4px;border:1px solid;font-size:13px;font-weight:bold;overflow:hidden;padding:3px 8px;display:flex;justify-content:space-between;align-items:center;z-index:1;box-sizing:border-box;}.zte-bar-wrap span{font-size:inherit;font-weight:inherit;}.zte-bar-up{color:#ff4c00;border-color:rgba(255,76,0,0.3);}.zte-bar-down{color:#0059fa;border-color:rgba(0,89,250,0.3);}.zte-bar-up::before{content:'';position:absolute;left:0;top:0;bottom:0;z-index:-1;background:rgba(255,76,0,0.12);width:var(--p-up,0%);transition:width 0.5s;}.zte-bar-down::before{content:'';position:absolute;left:0;top:0;bottom:0;z-index:-1;background:rgba(0,89,250,0.12);width:var(--p-down,0%);transition:width 0.5s;}#config-list.gege-list-container{contain:content!important;background-color:#ffffff!important;border-radius:8px!important;border:1px solid #e0e0e0!important;padding:20px 30px!important;box-shadow:0 2px 10px rgba(0,0,0,0.02)!important;margin-top:10px!important;}.gege-section{margin-bottom:10px;} .gege-section:last-child{margin-bottom:0;}.gege-list-container .config-title{font-size:16px!important;font-weight:bold!important;color:#333!important;margin:15px 0 10px 0!important;padding-bottom:5px!important;}.gege-list-container .gege-section:first-child .config-title{margin-top:0!important;}.gege-empty-state{color:#999!important;font-size:14px!important;padding:0 0 15px 5px!important;border-bottom:1px solid #f0f0f0!important;margin-bottom:5px!important;}.gege-list-item{background-color:transparent!important;border-bottom:1px solid #f0f0f0!important;padding:15px 10px!important;margin-bottom:0!important;border-radius:0!important;} .gege-list-item:last-child{border-bottom:none!important;}#zte-geek-board{contain:content;background-color:transparent!important;border-left:4px solid #0059fa!important;border-radius:0!important;padding:5px 0 5px 15px!important;margin:10px 0 15px 0!important;box-shadow:none!important;border-bottom:1px solid #f0f0f0!important;font-size:14px;display:flex;flex-direction:column;gap:6px;padding-bottom:15px!important;}#gege-global-overlay #zte-geek-board.geek-frozen-pane{position:sticky!important;top:0px!important;z-index:100!important;background-color:#f3f4f5!important;margin-top:0!important;padding-top:15px!important;box-shadow:0 10px 15px -3px rgba(0,0,0,0.05)!important;border-radius:0 0 8px 8px!important;}.gege-pin{cursor:pointer;font-size:11px;filter:grayscale(100%);opacity:0.5;transition:transform 0.2s;margin-left:2px;} .gege-pin.active{filter:none;opacity:1;transform:scale(1.1);}#gege-global-overlay{position:fixed;top:7.5%;right:0;bottom:0;background:#f3f4f5;z-index:9999;overflow-y:auto;padding-bottom:50px;left:0!important;border-radius:16px 16px 0 0;box-shadow:0 -5px 25px rgba(0,0,0,0.15);transition:top 0.3s ease;}@media (max-width: 768px){.geek-right-box:has(#gb-wan-zero-up),.geek-right-box:has(#gb-cur-up-vol){display:none!important}.gege-list-item{padding:12px 10px!important;position:relative!important}.config-item-box{flex-direction:column!important;padding-bottom:0!important}.config-item .info,.config-item .logo,.config-item .speed{width:100%!important;border:none!important;padding:0!important;position:static!important}.config-item .dev-intro{min-height:auto!important;justify-content:center!important;padding-right:90px!important}.config-item .logo{padding-bottom:4px!important}.config-item .info{flex-direction:column!important;margin:0 0 6px 0!important;gap:2px!important}.dev-ip{position:absolute!important;top:12px!important;right:10px!important;font-size:11px!important;background:rgba(0,89,250,0.08);color:#0059fa!important;padding:2px 6px!important;border-radius:4px;font-weight:bold;line-height:1.2;z-index:10;width:auto!important}.dev-number{width:auto!important;margin:0!important;font-size:11px!important}.gege-ratio-box{width:100%!important;margin-top:2px!important;margin-bottom:0!important}.gege-down-box{width:100%!important;margin-top:2px!important}#zte-geek-board{padding:8px!important;gap:0!important;font-size:11.5px!important}.geek-row{height:auto!important;flex-wrap:wrap!important;margin-bottom:4px!important;justify-content:flex-start!important;gap:2px 6px!important;line-height:1.3!important}.geek-label{width:auto!important;min-width:60px!important;font-size:11.5px!important;flex:0 0 auto!important}.geek-val-box{width:auto!important;flex:1 1 0%!important;display:flex!important;flex-wrap:wrap!important;margin-left:0!important;gap:2px 6px!important}.geek-fixed-width{width:auto!important}.geek-right-box{width:100%!important;flex:0 0 100%!important;text-align:left!important;font-size:11.5px!important;margin-top:2px!important;margin-left:0!important}.gege-list-container{padding:8px!important}.zte-enhance-speed{gap:4px!important}}`; document. head. appendChild(st); window.gegeRenderedMacs = new Set(); async function rSD() { if (window.__gIsF) return; window.__gIsF = !0; let n = performance.now(); try { let stk = window.__tpStok; if (!stk) { window.__gIsF = !1; return; // 还没到就等下一秒再试 } let apiUrl = `/stok=${stk}/ds`; let dW = null, dL = null; try { const resW = await fetch(apiUrl, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ network: { name: ["wan_status","wan_status_2"] }, method: "get" }) }); if (resW.ok) dW = await resW.json(); } catch(e) { console.warn("[TP] WAN口拉取失败", e); } try { const resL = await fetch(apiUrl, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ hosts_info: { table: "online_host" }, method: "get" }) }); if (resL.ok) dL = await resL.json(); } catch(e) { console.warn("[TP] 设备列表拉取失败", e); } if (!dL || dL.error_code !== 0) { window.__gIsF = !1; return; } // 设备列表是必须的 // WAN口容错:拉不到就用0 let curW = (dW && dW.network && dW.network.wan_status) ? dW.network.wan_status : {}; let cWU = (+curW.up_speed || 0) * 8000; // TP 的 up_speed 单位是 KBps,转成 bps 需要 * 8000 let cWD = (+curW.down_speed || 0) * 8000; // ... 后面的设备解析逻辑保持不变 ... // 容错双拨抓取(如果有万能马甲双WAN需求的话),若没有自然 fallback 到静默 false S.hasW2 = !!(dW?.network?.wan_status_2?.ipaddr && dW.network.wan_status_2.ipaddr !== "0.0.0.0"); if (S.hasW2) { let u2 = (+dW.network.wan_status_2.up_speed || 0) * 8, d2 = (+dW.network.wan_status_2.down_speed || 0) * 8; if (S.w2LT === undefined) S.w2LT = n; else if (S.w2U !== u2 || S.w2D !== d2) { let dt = n - S.w2LT; if (S.w2U > 0) S.w2TotUp += (S.w2U + u2) * dt * 0.0005; else if (u2 > 0) S.w2TotUp += u2 * 0.5 * CONFIG.wanRefreshInterval; if (S.w2D > 0) S.w2TotDn += (S.w2D + d2) * dt * 0.0005; else if (d2 > 0) S.w2TotDn += d2 * 0.5 * CONFIG.wanRefreshInterval; S.w2LT = n; } S.w2U = u2; S.w2D = d2; } let cSU = 0, cSD = 0, cI = Object.create(null); // 解衣:只剥 online_host 这种它自己唯一稳定的原生外皮 (dL.hosts_info?.online_host || []).forEach(obj => { let i = obj[Object.keys(obj)[0]]; if (i && i.mac) { let m = nM(i.mac), u = (+i.up_speed || 0) * 8, dn = (+i.down_speed || 0) * 8; cI[m] = { upRate: u, dnRate: dn, iface: i.type === '0' ? 'eth1' : (i.phy_mode === '7' ? 'wl1' : 'wl0'), offUp: 0, offDn: 0, onSec: +(i.online_time || 0), name: i.hostname || "未知设备", ip: i.ip || "" }; cSU += u; cSD += dn; } }); let ol = document.getElementById('gege-global-overlay'), cM = Object.keys(cI), iD = window.gegeForceUIRedraw || (cM.length !== window.gegeRenderedMacs.size); if (!iD && cM.length > 0) { for (let i = 0; i < cM.length; i++) { if (!window.gegeRenderedMacs.has(cM[i])) { iD = !0; break; } } } if (ol && ol.style.display === 'block' && (iD || !ol.querySelector('.gege-list-item'))) { bVD(ol, cI); window.gegeRenderedMacs = new Set(cM); window.gegeForceUIRedraw = !1; } let gDt = (S.lt !== 0) ? (n - S.lt) * 0.001 : 0; if (S.wLT === undefined) { S.wLT = n; } else if (cWU !== S.wInstUp || cWD !== S.wInstDn) { let wDt = n - S.wLT; if (S.wInstUp > 0) { S.wTotUp += (S.wInstUp + cWU) * wDt * 0.0005; } else if (cWU > 0) { let wEU = cWU * 0.5 * CONFIG.wanRefreshInterval; S.wTotUp += wEU; S.wZEU = (S.wZEU || 0) + wEU; S.wZEUC = (S.wZEUC || 0) + 1; } if (S.wInstDn > 0) { S.wTotDn += (S.wInstDn + cWD) * wDt * 0.0005; } else if (cWD > 0) { let wED = cWD * 0.5 * CONFIG.wanRefreshInterval; S.wTotDn += wED; S.wZED = (S.wZED || 0) + wED; S.wZEDC = (S.wZEDC || 0) + 1; } S.wLT = n; } for (const [m, cC] of Object.entries(cI)) { S.cls[m] ??= { upR: cC.upRate, dnR: cC.dnRate, lUT: n, intUp: _saved?.devices?.[m]?.integral_up || 0, intDn: _saved?.devices?.[m]?.integral_down || 0, onS: cC.onSec, lOS: cC.onSec, hU: [], hD: [] }; let cS = S.cls[m], dU = cC.offUp - cS.lU, dD = cC.offDn - cS.lD; if (dU < 0 || dD < 0) { if (dU < 0) { cS.uB += dU; cS.dpU = cS.lU; } if (dD < 0) { cS.dB += dD; cS.dpD = cS.lD; } cS.aR = !0; } else if (cS.aR) { if (dD > 2516582400 || dU > 671088640 || (cS.dpD && dD >= cS.dpD) || (cS.dpU && dU >= cS.dpU)) { cS.uB += dU; cS.dB += dD; cS.aR = !1; cS.dpU = 0; cS.dpD = 0; } } if (cS.lOS !== cC.onSec) { cS.onS = cC.onSec; cS.lOS = cC.onSec; } else { cS.onS = (cS.onS || cC.onSec || 0) + gDt; } if (cC.upRate !== cS.upR || cC.dnRate !== cS.dnR) { let ms = n - cS.lUT; if (cS.upR > 0) { cS.intUp += (cS.upR + cC.upRate) * ms * 0.0005; } else if (cC.upRate > 0) { let eU = cC.upRate * CONFIG.lanRefreshInterval * 0.5; cS.intUp += eU; cS.zEU = (cS.zEU || 0) + eU; cS.zUC = (cS.zUC || 0) + 1; } if (cS.dnR > 0) { cS.intDn += (cS.dnR + cC.dnRate) * ms * 0.0005; } else if (cC.dnRate > 0) { let eD = cC.dnRate * CONFIG.lanRefreshInterval * 0.5; cS.intDn += eD; cS.zED = (cS.zED || 0) + eD; cS.zDC = (cS.zDC || 0) + 1; } cS.upR = cC.upRate; cS.dnR = cC.dnRate; cS.lUT = n; } cS.lU = cC.offUp; cS.lD = cC.offDn; cS.hU.push(cC ? cC.upRate : 0); if (cS.hU.length > 120) cS.hU.shift(); cS.hD.push(cC ? cC.dnRate : 0); if (cS.hD.length > 120) cS.hD.shift(); } S.lt = n; S.wInstUp = cWU; S.wInstDn = cWD; rUI(cWU, cWD, cSU, cSD, cI); } catch (e) { console.error("[哥哥科技] 周期采样中断:", e); } finally { window.__gIsF = !1; } } const calcStageRatio = (W, L_int, L_hp) => { if (W === 0) return 1.0; let L_max = Math.max(L_int, L_hp); let L_min = Math.min(L_int, L_hp); let Gap = Math.abs(L_int - L_hp); if (L_int > 0.84 * W && L_hp > 0.75 * W && (L_max < 1.5 * W || Gap < 0.6 * W)) { return ((L_int + L_hp) / (2 * W)); } else if (L_min < W && W < L_max && L_max < 1.5 * W) { return L_max / W; } else { return (Math.abs(L_int - W) < Math.abs(L_hp - W) ? L_int : L_hp) / W; } }; function rUI(wU, wD, sU, sD, cI) { let LUp = 0, LDn = 0, hpU = 0, hpD = 0, curHpU = 0, curHpD = 0, cln = {}; for (let k in S.cls) { let s = S.cls[k]; let cC = cI[k]; let cU = s.intUp || 0; let cD = s.intDn || 0; let sessU = cU; let sessD = cD; LUp += s.intUp || 0; LDn += s.intDn || 0; hpU += sessU; hpD += sessD; if (cC) { curHpU += sessU; curHpD += sessD; } cln[k] = { up: cU, down: cD, integral_up: s.intUp || 0, integral_down: s.intDn || 0, status: s.aR ? "off" : (CONFIG.portMap[cC?.iface] || cC?.iface || "未知接口"), name: cC?.name || k, ip: cC?.ip || "", raw_up: cC?.offUp || 0, raw_down: cC?.offDn || 0 }; } if (typeof GM_setValue !== 'undefined') { try { GM_setValue('ha_snapshot', { timestamp: Date.now(), global: { wan_up: S.wTotUp, wan_down: S.wTotDn, lan_integral_up: LUp, lan_integral_down: LDn, lan_high_up: hpU, lan_high_down: hpD, }, devices: cln }); } catch(e) {console.warn(e);} } S.rTick = ((S.rTick || 0) + 1) & 15; if (S.rTick === 1 || !S.cRT) { S.aWu = (S.wTotUp - (S.lwTU || S.wTotUp)) / (CONFIG.wanRefreshInterval << 4); S.lwTU = S.wTotUp; S.aWd = (S.wTotDn - (S.lwTD || S.wTotDn)) / (CONFIG.wanRefreshInterval << 4); S.lwTD = S.wTotDn; if (S.hasW2) { let rU = S.w2TotUp > 0 ? (S.wTotUp / S.w2TotUp) : (S.wTotUp > 0 ? Infinity : 0), rD = S.w2TotDn > 0 ? (S.wTotDn / S.w2TotDn) : (S.wTotDn > 0 ? Infinity : 0); let fR = (r) => r === Infinity ? '∞' : (r > 1 ? r.toFixed(2) + 'x' : (r * 100).toPrecision(3) + '%'); S.cRT = `${fR(rU)}${fR(rD)}`; } else { let rUp = calcStageRatio(S.wTotUp, LUp, hpU), rDn = calcStageRatio(S.wTotDn, LDn, hpD); S.cRT = `${(rUp * 100).toFixed(2)}%${(rDn * 100).toFixed(2)}%`; } } let bd = document.getElementById('zte-geek-board'); if (!bd) { bd = document.createElement('div'); bd.id = 'zte-geek-board'; let layoutHtml = ''; if (CONFIG.uiLayout === 1) { // 紧凑版 (驾驶舱) layoutHtml = `
WAN口速率
|
0估算:
局域网代数和
实时占比: |
LAN:📌
WAN总计: |
${S.hasW2?'主次网比':'内外网比'}:
`; } else if (CONFIG.uiLayout === 2) { // 平铺版 (报表流) layoutHtml = `
WAN口速率
|
局域网代数和
实时占比: |
LAN:📌
WAN总计:
${S.hasW2?'主次网比':'内外网比'}:
`; } else { // 经典版 (0) layoutHtml = `
WAN口速率
|
局域网代数和
实时占比: |
LAN:📌
WAN: |
`; } bd.innerHTML = layoutHtml; let pinBtn = bd.querySelector('#gege-pin-btn'); if (pinBtn) { if (S.isPinned) { bd.classList.add('geek-frozen-pane'); pinBtn.classList.add('active'); } pinBtn.onclick = () => { S.isPinned = !S.isPinned; bd.classList.toggle('geek-frozen-pane', S.isPinned); pinBtn.classList.toggle('active', S.isPinned); }; } } let ol = document.getElementById('gege-global-overlay'), iPO = ol && ol.style.display === 'block', aC = iPO ? ol : document; requestAnimationFrame(() => { if (iPO) { let ac = document.getElementById('gege-board-anchor'); if (ac && bd.nextSibling !== ac) ac.parentNode.insertBefore(bd, ac); } else { let mn = document.querySelector('.el-table') || document.querySelector('.config-item')?.closest('div') || document.querySelector('.main-content'); if (mn && bd.parentNode !== mn.parentNode) mn.parentNode.insertBefore(bd, mn); } let oDC = Object.create(null); if (!iPO) { const M_RX = /([a-fA-F0-9]{2}[:-]){5}[a-fA-F0-9]{2}/; let aI = aC.querySelectorAll('.config-item'); for (let n of aI) { let mN = n.querySelector('.dev-number'), mM = mN ? mN.textContent.match(M_RX) : null; if (mM) { oDC[mM[0].toLowerCase().replace(/-/g, ':')] = n; } } } else { let gI = aC.querySelectorAll('.gege-list-item'); for (let n of gI) { let m = n.getAttribute('data-gege-mac'); if (m) oDC[m] = n; } } if (bd.parentNode) { const setText = (id, text) => { const el = bd.querySelector(id); if (el) el.textContent = text; }; let aW2U = S.hasW2 ? S.w2U : undefined, aW2D = S.hasW2 ? S.w2D : undefined, aW2TU = S.hasW2 ? S.w2TotUp : undefined, aW2TD = S.hasW2 ? S.w2TotDn : undefined; setText('#gb-wan-up-bytes', `🔼 ${fBy(wU + (aW2U||0))}`); setText('#gb-wan-down-bytes', `🔽 ${fBy(wD + (aW2D||0))}`); setText('#gb-wan-up-bps', `🔼 ${fB(wU)}`); setText('#gb-wan-down-bps', `🔽 ${fB(wD)}`); setText('#gb-lan-up-bytes', `🔼 ${fBy(sU)}`); setText('#gb-lan-down-bytes', `🔽 ${fBy(sD)}`); setText('#gb-perc-up', `🔼 ${wU>0?(sU*100/wU).toFixed(1):0.0}%`); setText('#gb-perc-down', `🔽 ${wD>0?(sD*100/wD).toFixed(1):0.0}%`); setText('#gb-lan-up-vol', `🔼 ${fV(LUp)}`); setText('#gb-lan-down-vol', `🔽 ${fV(LDn)}`); setText('#gb-wan-up-vol', `🔼 ${fV(S.wTotUp)}`); setText('#gb-wan-down-vol', `🔽 ${fV(S.wTotDn)}`); let pb = bd.querySelector('#gb-pwan-bps-container'), pv = bd.querySelector('#gb-pwan-vol-container'); if (aW2U !== undefined) { if (pb) { pb.style.display = 'inline'; setText('#gb-pwan-bps-up', '🔼 ' + fB(aW2U)); setText('#gb-pwan-bps-down', '🔽 ' + fB(aW2D)); } if (pv) { pv.style.display = 'inline'; setText('#gb-pwan-tot-up', '🔼 ' + fV(aW2TU)); setText('#gb-pwan-tot-down', '🔽 ' + fV(aW2TD)); } } else { if (pb) pb.style.display = 'none'; if (pv) pv.style.display = 'none'; } if (bd.querySelector('#gb-ratio-display')) { setText('#gb-cur-up-vol', `🔼 ${fV(curHpU)}`); setText('#gb-cur-down-vol', `🔽 ${fV(curHpD)}`); bd.querySelector('#gb-ratio-display').innerHTML = S.cRT; if (bd.querySelector('#gb-wan-zero-up')) { setText('#gb-wan-zero-up', !S.wZEU ? '' : fSV(S.wZEU)); setText('#gb-wan-zero-down', !S.wZED ? '' : fSV(S.wZED)); setText('#gb-wan-zero-up-cnt', S.wZEUC || 0); setText('#gb-wan-zero-down-cnt', S.wZEDC || 0); } } } for (let m in cI) { let it = oDC[m]; if (!it) continue; const cC = cI[m] || { upRate: 0, dnRate: 0, iface: "", offUp: 0, offDn: 0 }, cS = S.cls[m] || { intUp: 0, intDn: 0, onS: 0 }; let cache = it._gege || (it._gege = {}); let hqU = cS.intUp || 0; let hqD = cS.intDn || 0; let tN = cache.timeNode ??= it.querySelector('.gege-online-time'); if (tN && cS.onS > 0) tN.textContent = `在线:${fOT(cS.onS)}`; const dI = cache.devIntro ??= it.querySelector('.dev-intro'); const inf = cache.info ??= it.querySelector('.info'); if (dI && inf) { let rB = cache.rBox ??= dI.querySelector('.gege-ratio-box'); if (!rB) { let oRB = inf.querySelector('.gege-ratio-box'); if (oRB) oRB.remove(); rB = document.createElement('div'); rB.className = 'gege-ratio-box'; rB.style.cssText = 'margin-top: 4px; width: 95%; margin-bottom: 2px;'; rB.innerHTML = `
`; dI.appendChild(rB); cache.rBox = rB; } let bR = (hqU + hqD) > 0 ? (hqU * 100 / (hqU + hqD)) : 0, tC = "", tCol = "#0059fa"; if (CONFIG.calcMode === 1) { let rt = hqD > 0 ? (hqU / hqD) : (hqU > 0 ? Infinity : 0); if (rt > CONFIG.ratioExtremeUp) { tCol = '#ff4c00'; tC = (rt === Infinity ? '∞' : rt.toFixed(2)) + '⚠️'; } else if (rt > CONFIG.ratioWarnUp) { tCol = '#ff4c00'; tC = (rt * 100).toFixed(1) + '%'; } else if (rt > CONFIG.ratioExtremeDown) { tCol = '#0059fa'; tC = (rt * 100).toFixed(1) + '%'; } else { tCol = '#0059fa'; let rRt = hqU > 0 ? (hqD / hqU) : (hqD > 0 ? Infinity : 0); tC = (rRt === Infinity ? '∞' : rRt.toFixed(1)) + 'x'; } } else { tCol = bR > CONFIG.ratioThreshold ? '#ff4c00' : '#0059fa'; tC = bR.toFixed(1) + '%'; } (cache.rBoxPort ??= rB.querySelector('.v-port')).textContent = CONFIG.portMap[cC.iface] || cC.iface || "未知"; (cache.rBoxUp ??= rB.querySelector('.v-interval .c-up')).textContent = '' + fSV(hqU); (cache.rBoxDn ??= rB.querySelector('.v-interval .c-down')).textContent = '' + fSV(hqD); let rtP = cache.rtPct ??= rB.querySelector('.v-rt-pct'); rtP.textContent = tC; rtP.style.color = tCol; (cache.rBoxBar ??= rB.querySelector('.gege-ratio-bar-inner')).style.width = Math.min(bR, 100) + '%'; let ipNode = cache.ipNode ??= inf.querySelector('.dev-ip'); if (ipNode) { let zBadge = cache.zBadge ??= ipNode.querySelector('.gege-zero-badge'); if (!zBadge) { zBadge = document.createElement('span'); zBadge.className = 'gege-zero-badge gege-box'; ipNode.style.display = 'flex'; ipNode.style.justifyContent = 'space-between'; zBadge.style.cssText = 'color: #999; font-size: 11.5px; font-family: system-ui, sans-serif; margin-right: 5px;'; ipNode.appendChild(zBadge); cache.zBadge = zBadge; } zBadge.textContent = ((cS.zUC || 0) + (cS.zDC || 0)) < 6 ? "" : `[0估] ${!cS.zEU ? '' : fSV(cS.zEU)},${!cS.zED ? '' : fSV(cS.zED)}|${cS.zUC || 0},${cS.zDC || 0}`; } let bx = cache.upBox ??= inf.querySelector('.gege-up-box'); if (!bx || bx.querySelector('.t-row')) { if (bx) bx.remove(); let oUB = dI.querySelector('.gege-up-box'); if (oUB) oUB.remove(); bx = document.createElement('div'); bx.className = 'gege-up-box'; bx.style.cssText = 'display:flex; align-items:center; width:95%; margin-top:0px; margin-bottom:2px;'; bx.innerHTML = `
`; inf.appendChild(bx); cache.upBox = bx; } let p = hpU > 0 ? (hqU * 100 / hpU) : 0; (cache.upVol ??= bx.querySelector('.v-vol')).textContent = fV(cS.intUp); (cache.upPct ??= bx.querySelector('.v-pct')).textContent = p.toFixed(1) + '%'; (cache.upBar ??= bx.querySelector('.zte-thin-bar-inner')).style.width = Math.min(p, 100) + '%'; let dBx = cache.dBox ??= inf.querySelector('.gege-down-box'); if (!dBx || dBx.querySelector('.t-row')) { if (dBx) dBx.remove(); dBx = document.createElement('div'); dBx.className = 'gege-down-box'; dBx.style.cssText = 'display:flex; align-items:center; width:95%; margin-top:6px; margin-bottom:2px;'; dBx.innerHTML = `
`; inf.appendChild(dBx); cache.dBox = dBx; } let dp = LDn > 0 ? (cS.intDn * 100 / LDn) : 0; (cache.dBoxVol ??= dBx.querySelector('.v-vol')).textContent = fV(cS.intDn); (cache.dBoxPct ??= dBx.querySelector('.v-pct')).textContent = dp.toFixed(1) + '%'; (cache.dBoxBar ??= dBx.querySelector('.zte-thin-bar-inner')).style.width = Math.min(dp, 100) + '%'; } const sp = cache.speed ??= it.querySelector('.speed'); if (sp) { let enh = cache.enh ??= sp.querySelector('.zte-enhance-speed'); if (!enh) { sp.querySelectorAll('.connect-up, .connect-down').forEach(n => { n.style.display = 'none'; }); enh = document.createElement('div'); enh.className = 'zte-enhance-speed'; enh.innerHTML = `
`; sp.appendChild(enh); cache.enh = enh; } let pu = sU > 0 ? (cC.upRate * 100 / sU) : 0, pd = sD > 0 ? (cC.dnRate * 100 / sD) : 0, bU = cache.bU ??= enh.querySelector('.zte-bar-up'), bD = cache.bD ??= enh.querySelector('.zte-bar-down'); const SPRK = [' ', '▂', '▃', '▄', '▅', '▆', '▇', '█']; let clU = Math.max(...cS.hU, (S.aWu * 0.1) || 0, 512000); let clD = Math.max(...cS.hD, (S.aWd / 8) || 0); (cache.bUSpk ??= bU.querySelector('.v-spark')).textContent = cS.hU.slice(-60).map(v => SPRK[v < 73000 ? 0 : Math.min(7, Math.max(1, Math.floor((v / clU) * 7)))]).join(''); (cache.bDSpk ??= bD.querySelector('.v-spark')).textContent = cS.hD.slice(-60).map(v => SPRK[v < 1000000 ? 0 : Math.min(7, Math.max(1, Math.floor((v / clD) * 7)))]).join(''); bU.style.setProperty('--p-up', Math.min(pu, 100) + '%'); (cache.bUVal ??= bU.querySelector('.v-val')).textContent = `🔼 ${fBy(cC.upRate)}`; (cache.bUPct ??= bU.querySelector('.v-pct')).textContent = pu.toFixed(1) + '%'; bD.style.setProperty('--p-down', Math.min(pd, 100) + '%'); (cache.bDVal ??= bD.querySelector('.v-val')).textContent = `🔽 ${fBy(cC.dnRate)}`; (cache.bDPct ??= bD.querySelector('.v-pct')).textContent = pd.toFixed(1) + '%'; } } }); } async function bVD(ol, cI) { try { let h2 = [], h52 = [], h58 = [], hW = []; for (let m in cI) { let d = cI[m], tS = fOT(d.onSec), ifc = d.iface; let htm = `
${escapeHTML(d.ip)}
MAC:${m}
`; if (['wl0', '2.4G'].includes(ifc)) h2.push(htm); else if (['wlan5', 'wl1', 'wlan4', '5.2', '5.2G'].includes(ifc)) h52.push(htm); else if (ifc === 'wl2' || ifc === 'wlan2' || ifc === '5.8G' || (/w/i.test(ifc) && !/wan/i.test(ifc))) h58.push(htm); else hW.push(htm); } requestAnimationFrame(() => { ol.innerHTML = `
有线设备${(window.gegeHiddenDevices && Object.keys(window.gegeHiddenDevices).length > 0) ? '(哥哥科技:智能Mesh适配)' : ''}
${hW.join('')||'
没有连接设备
'}
无线设备(${S.is5G_149?'5.8GHz':'5.2GHz'})
${h52.join('')||'
没有连接设备
'}
无线设备(${S.is5G_149?'5.2GHz':'5.8GHz'})
${h58.join('')||'
没有连接设备
'}
无线设备(2.4GHz)
${h2.join('')||'
没有连接设备
'}
“在一个文明社会,干净的、不被监视与吸血的网络,是我们每个人的基本权利。”
本交互式程序基于 GNU Affero GPL v3.0 协议开源,按“原样 (AS IS)”提供,不对其适用性、稳定性、精密度或任何商业场景合规性作任何明示或暗示的担保。
根据 AGPL-3.0 第 5(d) 及 7(b) 条规定,基于本程序的任何修改均不得移除或篡改本界面的署名与法律声明。保留此界面是使用本软件代码的合法性的前置条件。
Bro-Stat 增强组件 Copyright © 2026 哥哥科技 (BroTech) | All Rights Reserved  点此分享
`; });} catch (e) { requestAnimationFrame(() => { ol.innerHTML = `
数据渲染失败: ${escapeHTML(e.message)}
`; }); } } window.createGegeFloatingBtn = function () { if (document.getElementById('gege-floating-btn')) return; let b = document.createElement('div'); b.id = 'gege-floating-btn'; b.innerHTML = '🛸'; b.style.cssText = 'position: fixed; top: 20px; right: 30%; width: 50px; height: 50px; background: linear-gradient(135deg, #0059fa, #00c6ff); color: white; border-radius: 50%; display: flex; align-items: center; justify-content: center; font-size: 48px; box-shadow: 0 4px 15px rgba(0,89,250,0.5); cursor: pointer; z-index: 99999; transition: transform 0.3s ease; user-select: none;'; b. onmouseover = () => { b.style.transform = 'scale(1.1) rotate(15deg)'; }; b.onmouseout = () => { b.style.transform = 'scale(1) rotate(0deg)'; }; b.onclick = () => window.gegeTogglePanel(); document.body.appendChild(b); }; window.gegeTogglePanel = function (fS = null) { let o = document.getElementById('gege-global-overlay'); let tS = fS !== null ? fS : !(o && o.style.display === 'block'); if (!tS) { if (o) o.style.display = 'none'; return; } if (!o) { o = document.createElement('div'); o.id = 'gege-global-overlay'; document.body.appendChild(o); } o.style.display = 'block'; if (!window.gegeBActivated) { window.gegeBActivated = !0; clearTimeout(window.gegeMasterTimer); window.gegeMasterTimer = setInterval(rSD, CONFIG.lanRefreshInterval * 1000); } bVD(o, Object.create(null)).then(() => rSD()); }; window.gegeBActivated = !1; window.gegeEngineRunning = !1; window.gegeLastDevCount = -1; window.gegeLastMeshDevCount = -1; window.gegeHiddenDevices = {}; window.gegeTimerStarted = !1; window.gegeSyncAnchor = 0; window.gegeTickCount = 0; window.gegeMasterTimer = null; window.startGegePrecisionEngine = function () { if (window.gegeTimerStarted || window.gegeBActivated) return; window.gegeTimerStarted = !0; window.gegeSyncAnchor = performance.now(); window.gegeTickCount = 0; window.scheduleNextGegeTick(); }; window.scheduleNextGegeTick = function () { if (window.gegeBActivated) return; window.gegeTickCount++; let dl = (window.gegeSyncAnchor + window.gegeTickCount * 3000) - performance.now(); if (dl < 0) { window.gegeSyncAnchor = performance.now(); window.gegeTickCount = 0; dl = 3000; } window.gegeMasterTimer = setTimeout(() => { rSD().finally(() => { window.scheduleNextGegeTick(); }); }, dl); }; const tKA = () => { let i = document.createElement('iframe'); i.id = 'gege-keepalive-iframe'; i.style.display = 'none'; const p = ["/#/sys", "/#/app", "/#/wlan/"]; i.src = `${window.location.origin}${p[Math.floor(Math.random()*p.length)]}`; let z = document.getElementById('gege-keepalive-iframe'); if (z) z.remove(); document.body.appendChild(i); setTimeout(() => { if (i.parentNode) { i.src = 'about:blank'; i.remove(); } }, 12000); }; setTimeout(tKA, 2000); setInterval(tKA, 720000); const _initUI = () => { if (CONFIG.injectMode === 3 || (CONFIG.injectMode === 1 && +(window.location.hostname.slice(window.location.hostname.lastIndexOf('.') + 1)) < 6)) { if (window.createGegeFloatingBtn) window.createGegeFloatingBtn(); } }; if (document.readyState === 'complete') _initUI(); else window.addEventListener('load', _initUI); })();