// ==UserScript== // @name B仔 Css隐藏规则日志 // @namespace http://bzbrowser.com/ // @version 1.1 // @license MIT // @description 检测哪些Css规则在B仔浏览器上生效,并输出匹配日志。 // @author Copilot & Grok & Gemini // @run-at document-end // @match *://*/* // @icon  // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_getValue // ==/UserScript== (function() { 'use strict'; const currentDomain = location.hostname || 'unknown'; const CONFIG = { STORAGE: { SETTINGS: 'bz_css_configs', ENABLED: 'floatingButtonEnabled' }, CSS_PATH: '/adb.css-bzbrowser', IDLE_TIME: 3000, BATCH_SIZE: 100 }; let settings = typeof GM_getValue !== 'undefined' ? GM_getValue(CONFIG.STORAGE.SETTINGS, { floatEnabled: true }) : { floatEnabled: true }; let activeRulesCache = []; let shadowRoot = null, container = null; const UI_CSS = ` .mask { position:fixed; top:0; left:0; width:100%; height:100%; background:rgba(0,0,0,0.3); backdrop-filter:blur(12px); -webkit-backdrop-filter:blur(12px); z-index:2147483646; display:flex; align-items:center; justify-content:center; animation: fade-in 0.4s cubic-bezier(0.22, 1, 0.36, 1); pointer-events: auto; touch-action: none; } .panel { background:rgba(255,255,255,0.9); border:1px solid rgba(255,255,255,0.4); border-radius:28px; box-shadow:0 25px 50px -12px rgba(0,0,0,0.25); padding:24px; display:flex; flex-direction:column; gap:14px; width:92%; max-width:450px; max-height:82vh; font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Helvetica,Arial,sans-serif; box-sizing:border-box; position:relative; overflow:hidden; touch-action: auto; } .title { margin:0 0 4px 0; font-size:19px; font-weight:700; color:#1d1d1f; text-align:center; letter-spacing:-0.02em; } .list-container { flex:1; overflow-y:auto; display:flex; flex-direction:column; gap:12px; padding-right:4px; scroll-behavior: smooth; } .list-container::-webkit-scrollbar { width:4px; } .list-container::-webkit-scrollbar-thumb { background:rgba(0,0,0,0.1); border-radius:10px; } .rule-card { background:rgba(255,255,255,0.6); border-radius:16px; padding:14px; cursor:pointer; transition:all 0.4s cubic-bezier(0.22, 1, 0.36, 1); border:1px solid rgba(0,0,0,0.05); position:relative; display:flex; flex-direction:column; } .rule-card:hover { transform: translateY(-2px); background:#fff; box-shadow: 0 10px 20px rgba(0,0,0,0.05); border-color: rgba(0,122,255,0.2); } .rule-line { font-family:"SF Mono",SFMono-Regular,Consolas,monospace; font-size:13px; white-space:nowrap; overflow-x:auto; padding-bottom:8px; scrollbar-width: none; line-height:1.4; } .rule-line::-webkit-scrollbar { display: none; } .rule-badge { position:absolute; bottom:8px; right:10px; background:rgba(0,122,255,0.08); color:#007AFF; font-size:11px; padding:2px 8px; border-radius:8px; font-weight:600; pointer-events:none; transition: opacity 0.3s; } .rule-content { margin-top:10px; display:none; flex-direction:column; gap:10px; border-top:1px solid rgba(0,0,0,0.06); padding-top:14px; animation: slide-up 0.4s cubic-bezier(0.22, 1, 0.36, 1); } .rule-card.expanded .rule-content { display:flex; } .rule-card.expanded .rule-badge { opacity: 0; } .hl-domain { color: #ff8c00; font-weight: 600;} .hl-sep { color: #007bff; font-weight: 700; } .hl-selector { color: #808080; } .hl-url { color: #ff0000; font-weight: 600; } .hl-pseudo { color: #d197d9; font-weight: 600; } .hl-paren { color: #deb887; } .btn-group { display:grid; grid-template-columns: 1fr 1fr; gap:10px; } button { border:none; border-radius:14px; padding:10px; cursor:pointer; font-size:13px; font-weight:600; transition:all 0.3s cubic-bezier(0.22, 1, 0.36, 1); background:#f2f2f7; color:#1d1d1f; } button:active { transform: scale(0.96); } button.primary { background:#007AFF; color:#fff; width: 100%; margin-top: 6px; box-shadow: 0 4px 12px rgba(0,122,255,0.3); } button.copy-btn { background:#007AFF; color:white; } button.allow-btn { background:#ff9500; color:white; } .auth-settings-panel { position:relative; background:rgba(255,255,255,0.85); border-radius:28px; box-shadow:0 25px 50px -12px rgba(0,0,0,0.25); padding:24px; display:flex; flex-direction:column; gap:12px; width:300px; animation: slide-in 0.4s cubic-bezier(0.16, 1, 0.3, 1); border:1px solid rgba(255,255,255,0.4); touch-action: auto; } .card-slider-wrapper { position:relative; width:100%; overflow:hidden; margin-bottom:4px; border-radius:18px; } .card-slider { display:flex; transition: transform 0.3s cubic-bezier(0.4, 0, 0.2, 1); } .result-card-mini { min-width:100%; box-sizing:border-box; background:rgba(255,255,255,0.5); border-radius:18px; padding:18px; display:flex; flex-direction:column; align-items:center; border:1px solid rgba(0,0,0,0.05); } .mini-title { font-size:14px; font-weight:800; color:#007AFF; margin-bottom:4px; } .mini-desc { font-size:12px; color:#666; text-align:center; line-height:1.4; } .dots { display:flex; justify-content:center; gap:5px; margin-top:8px; } .dot { width:6px; height:6px; border-radius:50%; background:rgba(0,0,0,0.1); transition: 0.3s; } .dot.active { background:#007AFF; width:12px; border-radius:3px; } .auth-settings-panel button.menu-btn { border:none; border-radius:14px; padding:14px; cursor:pointer; font-size:14px; transition: all 0.2s; display:flex; align-items:center; justify-content:space-between; font-weight:600; background:#fff; color:#007AFF; box-shadow:0 4px 12px rgba(0,122,255,0.1); width:100%; box-sizing:border-box; } .btn-close { margin-top:6px; background:none !important; color:#888 !important; box-shadow:none !important; justify-content:center !important; } @media (prefers-color-scheme: dark) { .panel { background:rgba(28,28,30,0.9); border-color:rgba(255,255,255,0.1); color:#fff; } .title { color:#fff; } .rule-card { background:rgba(44,44,46,0.6); border-color:rgba(255,255,255,0.05); } .rule-card:hover { background:rgba(58,58,60,0.8); border-color:rgba(0,122,255,0.4); } .rule-badge { background:rgba(255,255,255,0.1); color:#0a84ff; } .hl-selector { color: #d1d1d6; } button { background:#3a3a3c; color:#fff; } .auth-settings-panel { background:rgba(30,30,30,0.85); border-color:rgba(255,255,255,0.1); } .result-card-mini { background:rgba(255,255,255,0.05); border-color:rgba(255,255,255,0.05); } .auth-settings-panel button.menu-btn { background:#2c2c2e; color:#0A84FF; } .mini-desc { color:#aaa; } } @keyframes fade-in { from { opacity:0; } to { opacity:1; } } @keyframes slide-up { from { opacity:0; transform:translateY(10px); } to { opacity:1; transform:translateY(0); } } @keyframes slide-in { from { opacity:0; transform: translateY(20px) scale(0.95); } to { opacity:1; transform: translateY(0) scale(1); } } .float-btn { position:fixed; right:0; top:60%; width:36px; height:54px; background:rgba(255, 255, 255, 0.3); border-radius:20px 0 0 20px; z-index:2147483645; display:flex; align-items:center; justify-content:center; cursor:pointer; box-shadow:0 8px 32px rgba(0, 0, 0, 0.15); backdrop-filter:blur(16px); -webkit-backdrop-filter:blur(16px); transition: all 0.5s cubic-bezier(0.22, 1, 0.36, 1); border:1px solid rgba(255,255,255,0.4); border-right:none; color:#1d1d1f; font-size:13px; font-weight:700; user-select:none; touch-action:none; pointer-events: auto; } .float-btn.idle { opacity:0.4; transform:translateX(18px); filter: grayscale(0.5); } .pop { animation: popIn 0.5s cubic-bezier(0.34, 1.56, 0.64, 1); } @keyframes popIn { from { transform:scale(0.8); opacity:0; } to { transform:scale(1); opacity:1; } } `; const ensureShadow = () => { if (shadowRoot) return; container = document.createElement('div'); container.id = 'bz-css-logger-manager'; container.style.cssText = 'position:fixed;top:0;left:0;z-index:2147483647;pointer-events:none;'; document.documentElement.appendChild(container); shadowRoot = container.attachShadow({ mode: 'closed' }); const style = document.createElement('style'); style.textContent = UI_CSS; shadowRoot.appendChild(style); }; const showToast = (msg) => { if (window.bzhome?.toast) { window.bzhome.toast(msg); return; } ensureShadow(); const toast = document.createElement('div'); toast.style.cssText = 'position:fixed;top:80vh;left:50vw;transform:translate(-50%,-50%);background:rgba(255,255,255,0.4);backdrop-filter:blur(16px);-webkit-backdrop-filter:blur(16px);padding:12px 24px;border-radius:20px;box-shadow:0 8px 32px rgba(0,0,0,0.1);color:#1C2526;font-size:14px;font-weight:500;z-index:2147483647;opacity:0;transition:opacity 0.3s;pointer-events:none;border:1px solid rgba(255,255,255,0.2);'; toast.textContent = msg; shadowRoot.appendChild(toast); requestAnimationFrame(() => toast.style.opacity = '1'); setTimeout(() => { toast.style.opacity = '0'; setTimeout(() => toast.remove(), 300); }, 2000); }; const Core = { splitSelectors(cssText) { const selectors = new Set(); let current = '', inBlock = false, bracketDepth = 0, parenDepth = 0, inQuote = false, quoteChar = null; for (let i = 0; i < cssText.length; i++) { const char = cssText[i]; if (inQuote) { current += char; if (char === quoteChar) inQuote = false; continue; } if (char === '"' || char === "'") { inQuote = true; quoteChar = char; current += char; continue; } if (inBlock) { if (char === '}') inBlock = false; continue; } if (char === '[') bracketDepth++; if (char === ']') bracketDepth--; if (char === '(') parenDepth++; if (char === ')') parenDepth--; if (char === '{' && bracketDepth === 0 && parenDepth === 0 && !inQuote) { if (current.trim()) selectors.add(current.trim()); current = ''; inBlock = true; continue; } if (char === ',' && bracketDepth === 0 && parenDepth === 0 && !inQuote && !inBlock) { if (current.trim()) selectors.add(current.trim()); current = ''; } else { current += char; } } return Array.from(selectors).filter(s => s && !s.includes('!important') && !s.startsWith('@')); }, async checkActiveSelectors(cssText) { const selectors = this.splitSelectors(cssText); const activeMap = new Map(); for (let i = 0; i < selectors.length; i += CONFIG.BATCH_SIZE) { const batch = selectors.slice(i, i + CONFIG.BATCH_SIZE); batch.forEach(selector => { try { const count = document.querySelectorAll(selector).length; if (count > 0 && !activeMap.has(selector)) activeMap.set(selector, count); } catch (e) {} }); await new Promise(r => setTimeout(r, 0)); } return Array.from(activeMap.entries()).map(([selector, count]) => ({ selector, count })); } }; const UI = { highlightAdRule(rule) { const match = rule.match(/^(.*?)(###?)(.*)$/); if (!match) return `${rule}`; let rest = match[3].replace(/("(.*?)")/g, '"$2"').replace(/(:(?:has|not|is|where|nth-child|hover|focus|active))(\(.*?\))?/g, '$1$2'); return `${match[1]}${match[2]}${rest}`; }, copyRule(selector, isAllow) { let text = `${location.hostname}##${selector}`; if (isAllow) text = text.replace('###', '#@##').replace('##', '#@#'); navigator.clipboard.writeText(text).then(() => showToast('已复制规则')); }, showPanel(activeRules) { ensureShadow(); const mask = document.createElement('div'); mask.className = 'mask'; mask.onclick = (e) => { if (e.target === mask) mask.remove(); }; const panel = document.createElement('div'); panel.className = 'panel pop'; panel.onclick = (e) => e.stopPropagation(); const title = document.createElement('div'); title.className = 'title'; title.textContent = `生效规则 (${activeRules.length})`; const list = document.createElement('div'); list.className = 'list-container'; activeRules.forEach(r => { const card = document.createElement('div'); card.className = 'rule-card'; const fullRule = `${location.hostname}##${r.selector}`; card.innerHTML = `
${this.highlightAdRule(fullRule)}
${r.count}
`; card.onclick = (e) => { if (e.target.tagName !== 'BUTTON') card.classList.toggle('expanded'); }; card.querySelector('.copy-btn').onclick = () => this.copyRule(r.selector, false); card.querySelector('.allow-btn').onclick = () => this.copyRule(r.selector, true); list.appendChild(card); }); const closeBtn = document.createElement('button'); closeBtn.className = 'primary'; closeBtn.textContent = '关闭界面'; closeBtn.onclick = () => mask.remove(); panel.append(title, list, closeBtn); mask.appendChild(panel); shadowRoot.appendChild(mask); } }; const renderSettings = () => { if (shadowRoot?.querySelector('.auth-settings-mask')) return; ensureShadow(); const mask = document.createElement('div'); mask.className = 'mask auth-settings-mask'; mask.addEventListener('touchmove', (e) => e.preventDefault(), { passive: false }); const panel = document.createElement('div'); panel.className = 'auth-settings-panel'; panel.onclick = (e) => e.stopPropagation(); const sliderWrapper = document.createElement('div'); sliderWrapper.className = 'card-slider-wrapper'; const slider = document.createElement('div'); slider.className = 'card-slider'; const dotsCont = document.createElement('div'); dotsCont.className = 'dots'; let currIdx = 0; const updateSlider = (index) => { currIdx = index; slider.style.transform = `translateX(-${currIdx * 100}%)`; Array.from(dotsCont.children).forEach((d, i) => d.className = 'dot' + (i === currIdx ? ' active' : '')); }; if (activeRulesCache.length === 0) { const emptyCard = document.createElement('div'); emptyCard.className = 'result-card-mini'; emptyCard.innerHTML = `
暂无缓存数据
请先运行检测或在当前页面扫描
`; slider.appendChild(emptyCard); } else { activeRulesCache.forEach((r, i) => { const card = document.createElement('div'); card.className = 'result-card-mini'; card.style.minWidth = '100%'; card.innerHTML = `

${currentDomain}

规则 #${i + 1} (匹配: ${r.count})
${r.selector}
`; card.querySelector('.mini-copy-btn').onclick = () => UI.copyRule(r.selector, false); card.querySelector('.mini-allow-btn').onclick = () => UI.copyRule(r.selector, true); slider.appendChild(card); const dot = document.createElement('div'); dot.className = 'dot' + (i === 0 ? ' active' : ''); dot.style.cursor = 'pointer'; dot.onclick = () => updateSlider(i); dotsCont.appendChild(dot); }); } let startX = 0; sliderWrapper.ontouchstart = (e) => startX = e.touches[0].clientX; sliderWrapper.ontouchend = (e) => { if (activeRulesCache.length <= 1) return; let diff = startX - e.changedTouches[0].clientX; if (Math.abs(diff) > 50) { if (diff > 0 && currIdx < activeRulesCache.length - 1) updateSlider(currIdx + 1); else if (diff < 0 && currIdx > 0) updateSlider(currIdx - 1); } }; const btnCheck = document.createElement('button'); btnCheck.className = 'menu-btn'; btnCheck.innerHTML = `立即检测当前页面 🔍`; btnCheck.onclick = () => { mask.remove(); runCheck(); }; const btnFloat = document.createElement('button'); btnFloat.className = 'menu-btn'; btnFloat.innerHTML = `右侧悬浮按钮 ${settings.floatEnabled ? 'ON' : 'OFF'}`; btnFloat.onclick = () => { settings.floatEnabled = !settings.floatEnabled; GM_setValue(CONFIG.STORAGE.SETTINGS, settings); GM_setValue(CONFIG.STORAGE.ENABLED, settings.floatEnabled); btnFloat.querySelector('small').textContent = settings.floatEnabled ? 'ON' : 'OFF'; showToast(settings.floatEnabled ? '悬浮按钮已开启' : '悬浮按钮已关闭'); }; const close = document.createElement('button'); close.className = 'btn-close'; close.textContent = '保存并返回'; close.onclick = () => { if (GM_getValue(CONFIG.STORAGE.ENABLED) !== settings.floatEnabled) location.reload(); else mask.remove(); }; sliderWrapper.appendChild(slider); panel.append(sliderWrapper, dotsCont, btnCheck, btnFloat, close); mask.appendChild(panel); mask.onclick = () => mask.remove(); shadowRoot.appendChild(mask); }; const runCheck = async () => { showToast('正在扫描...'); const cssUrl = `${window.location.protocol}//${window.location.hostname}${CONFIG.CSS_PATH}`; try { const resp = await fetch(cssUrl, { cache: 'no-cache' }); if (!resp.ok) throw new Error(); const cssText = await resp.text(); const activeRules = await Core.checkActiveSelectors(cssText); activeRulesCache = activeRules; if (activeRules.length > 0) UI.showPanel(activeRules); else showToast('未发现生效规则'); } catch (e) { showToast('获取规则失败'); } }; const initFloatBtn = () => { ensureShadow(); const floatBtn = document.createElement('div'); floatBtn.className = 'float-btn'; floatBtn.innerHTML = `CSS`; let idleTimer; const resetIdle = () => { floatBtn.classList.remove('idle'); clearTimeout(idleTimer); idleTimer = setTimeout(() => { if (!shadowRoot.querySelector('.mask')) floatBtn.classList.add('idle'); }, CONFIG.IDLE_TIME); }; let isDragging = false, startY, startTop, moved = false; floatBtn.ontouchstart = (e) => { isDragging = true; moved = false; startY = e.touches[0].clientY; startTop = floatBtn.offsetTop; floatBtn.style.transition = 'none'; resetIdle(); }; window.addEventListener('touchmove', (e) => { if (!isDragging) return; moved = (Math.abs(e.touches[0].clientY - startY) > 5); floatBtn.style.top = (startTop + e.touches[0].clientY - startY) + 'px'; }, { passive: false }); window.addEventListener('touchend', () => { if (!isDragging) return; isDragging = false; floatBtn.style.transition = 'all 0.4s cubic-bezier(0.2, 0.8, 0.2, 1)'; floatBtn.style.top = Math.max(50, Math.min(window.innerHeight - 50, floatBtn.offsetTop)) + 'px'; floatBtn.style.right = '0px'; resetIdle(); }); floatBtn.onclick = () => { if (!moved) runCheck(); }; shadowRoot.appendChild(floatBtn); resetIdle(); }; if (typeof GM_registerMenuCommand !== 'undefined') { GM_registerMenuCommand('CSS 隐藏规则设置 🛠️', renderSettings); } if (GM_getValue(CONFIG.STORAGE.ENABLED, true)) initFloatBtn(); })();