// ==UserScript== // @name 移除广告内嵌脚本 // @namespace https://greasyfork.org/zh-CN/users/1373566 // @version 1.6 // @license MIT // @description 这是一个由AI生成的脚本,通过关键词匹配来移除网页中的内嵌广告脚本。不能保证100%成功,可以在脚本菜单中管理排除的网页和关键词,脚本已经内置一些常见的关键词,若还有广告,可以自行添加Math.random,platform,navigator,new Function,new Date()尝试去除。 // @author copilot & cheatgpt // @homepageURL https://scriptcat.org/zh-CN/script-show-page/2796 // @match http*://*/* // @exclude *://*.baidu.*/* // @exclude *://*.bing.*/* // @exclude *://*.douban.com/* // @exclude *://*.github.com/* // @exclude *://*.google.*/* // @exclude *://*.ifeng.com/* // @exclude *://*.iqiyi.com/* // @exclude *://*.mgtv.com/* // @exclude *://*.pptv.com/* // @exclude *://*.qq.com/* // @exclude *://*.sina.com.cn/* // @exclude *://*.sohu.com/* // @exclude *://*.v.qq.com/* // @exclude *://*.yandex.*/* // @exclude *://github.com/* // @exclude *://greasyfork.org/* // @exclude *://m.douban.com/* // @exclude *://scriptcat.org/* // @exclude *://twitter.com/* // @exclude *://x.com/* // @icon  // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @run-at document-start // ==/UserScript== (function() { 'use strict'; const REMOVE_AD_SCRIPTS_KEYWORDS_KEY = 'removeAdScriptsKeywords'; const EXCLUDE_SITES_KEY = 'excludeSites'; const SPOOF_UA_SITES_KEY = 'spoofUASites'; const DEFAULT_KEYWORDS = [ 'htmlAds', '_ffc1();', '_ffc2();', '_ffc3();', '_ffc4();', '_ffp();', 'kbbaidu1();', 'kbbaidu2();', 'kbbaidu3();', 'html5player.checkVideoAds', 'ads_codes', '{return void 0!==b[a]?b[a]:a}).join("")}', '${scripts[randomIndex]}', '${scripts[Math.random()', '"https://"+Date.parse(new Date())+', '"https://"+(new Date().getDate())+', 'https://hongosi.xn--', 'https://{randomstr}.', 'new Function(t)()', 'new Function(b)()', 'new Function(c)()', 'new Function(t);', 'new Function(b);', 'new Function(c);', 'new Function(\'d\',e)', 'new Function(document[', 'new Function(function(p,a,c,k,e,d)', 'function a(a){', 'function b(b){', 'function c(c){', 'function updateCarousel()', 'Math.floor(2147483648 * Math.random());', 'Math.floor(Math.random()*url.length)', 'Math.floor(Math.random() * urls.length)', 'new Date()[\'getTime\']()', 'newDate=new window', 'Math.floor(((new Date()).getTime()', '&&navigator[', '=navigator;', 'navigator.platform){setTimeout(function', 'disableDebugger', 'blockDeveloperTools', '["Date"]())[\'getTime\']()', '\'+\'s\'+\'c\'+\'ri\'+\'pt\'+\'>\');', '<\\/\'+\'s\'+\'c\'+\'ri\'+\'pt\'+\'>\');', 'class=\\"zdhf\\"', '(\'#htmlContenthtml\').html', 'D.createElement(\'span\');', 'class=\\"app_tj\\"', 'window.$m(', 'jsjiami.com.v4', 'histats.com', 'hm.baidu.com', 'adbyunion', '{win:false,mac:false,xll:false}', 'mainCell:".bd",effect:"leftLoop"', '/invoke.js">', 'function|getDate', 'parseInt(Math[\'random\']', 'pc.stgowan.com' ]; const UI_CSS = ` .mask { position:fixed; top:0; left:0; width:100%; height:100%; background:rgba(0,0,0,0.3); backdrop-filter:blur(8px); z-index:2147483646; display:flex; align-items:center; justify-content:center; animation: fade-in 0.3s ease; pointer-events: auto; } .panel { background:#fff; border-radius:24px; box-shadow:0 20px 40px rgba(0,0,0,0.2); padding:24px; display:flex; flex-direction:column; gap:12px; width:90%; max-width:400px; font-family:system-ui,-apple-system,sans-serif; box-sizing:border-box; position:relative; } .title { margin:0 0 5px 0; font-size:16px; font-weight:700; color:#1a1a1a; text-align:center; white-space:nowrap; overflow:hidden; text-overflow:ellipsis; } .btn-group { display:grid; grid-template-columns: 1fr 1fr; gap:8px; } button { border:none; border-radius:12px; padding:10px; cursor:pointer; font-size:12px; font-weight:600; transition:all 0.2s; background:#f0f2f5; color:#444; display:flex; align-items:center; justify-content:center; } button:hover { background:#e4e6e9; } button:active { transform:scale(0.95); } button.primary { background:#007AFF; color:#fff; } button.primary:hover { background:#0063cc; } button.danger { background:#ff4d4f; color:#fff; } button.danger:hover { background:#d9363e; } textarea { width:100%; height:160px; border:1px solid #ddd; border-radius:12px; padding:12px; font-family:monospace; font-size:12px; resize:none; box-sizing:border-box; outline:none; line-height:1.5; } textarea:focus { border-color:#007AFF; box-shadow:0 0 0 3px rgba(0,122,255,0.1); } select { width:100%; padding:10px; border-radius:12px; border:1px solid #ddd; outline:none; font-size:13px; } .footer { display:flex; flex-direction:column; gap:5px; margin-top:5px; } @media (prefers-color-scheme: dark) { .panel { background:#1c1c1e; color:#fff; } .title { color:#fff; } button { background:#2c2c2e; color:#ccc; } button:hover { background:#3a3a3c; } textarea { background:#2c2c2e; border-color:#444; color:#eee; } select { background:#2c2c2e; border-color:#444; color:#eee; } } @keyframes fade-in { from { opacity:0; } to { opacity:1; } }`; const TOTAL_REMOVED_KEY = 'totalRemovedCount'; let totalRemoved = GM_getValue(TOTAL_REMOVED_KEY, 0); let keywords = GM_getValue(REMOVE_AD_SCRIPTS_KEYWORDS_KEY, []); let spoofUAList = GM_getValue(SPOOF_UA_SITES_KEY, []); let excludeList = GM_getValue(EXCLUDE_SITES_KEY, []); let removedScriptsInfo = []; let shadowRoot = null; const ensureShadow = () => { if (shadowRoot) return; const container = document.createElement('div'); container.id = 'ad-remover-settings-container'; container.style.cssText = 'position:absolute;top:0;left:0;z-index:2147483647;'; document.documentElement.appendChild(container); shadowRoot = container.attachShadow({ mode: 'closed' }); const style = document.createElement('style'); style.textContent = UI_CSS; shadowRoot.appendChild(style); }; const toggleSecurePicking = (state) => { const UI_ID = 'ad-remover-settings-container'; const html = document.documentElement; if (!window._pickerHandler) { window._pickerHandler = (e) => { const path = e.composedPath(); if (!path.some(el => el.id === UI_ID)) { e.stopPropagation(); e.preventDefault(); } }; } if (state) { document.designMode = 'on'; html.style.pointerEvents = 'none'; const ui = document.getElementById(UI_ID); if (ui) ui.style.pointerEvents = 'auto'; window.addEventListener('click', window._pickerHandler, true); window.onbeforeunload = () => "正在编辑,禁止离开"; } else { document.designMode = 'off'; html.style.pointerEvents = 'auto'; window.removeEventListener('click', window._pickerHandler, true); window.onbeforeunload = null; removeSpecificScript(); } }; const spoofPlatform = (p = 'Mac') => { const s = new Proxy(navigator, { get(t, k) { if (k === 'platform') return p; let v = Reflect.get(t, k); return typeof v === 'function' ? v.bind(t) : v; }, getOwnPropertyDescriptor: (t, k) => k === 'platform' ? { value: p, writable: false, configurable: true, enumerable: true } : Object.getOwnPropertyDescriptor(t, k) }); const f = () => { try { if (Object.getOwnPropertyDescriptor(navigator, 'platform')) { navigator.__defineGetter__('platform', () => p); } else { Object.defineProperty(navigator, 'platform', { get: () => p, configurable: true, enumerable: true }); } Object.defineProperty(window, 'navigator', { value: s, writable: false, configurable: true }); Object.defineProperty(Navigator.prototype, 'platform', { get: () => p, configurable: true, enumerable: true }); Object.freeze(window.navigator); } catch (e) {} }; const i = String.prototype.indexOf; String.prototype.indexOf = function(v) { if (this === p) { if (['Win', 'Linux', 'X11'].includes(v)) return -1; if (v === p) return 0; } return i.apply(this, arguments); }; f(); updateCount(1); if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', () => { try { Object.defineProperty(window, 'navigator', { value: s, writable: false }); } catch(e){} }, { once: true }); } }; const saveKeywords = () => GM_setValue(REMOVE_AD_SCRIPTS_KEYWORDS_KEY, keywords); const saveExcludeList = () => GM_setValue(EXCLUDE_SITES_KEY, excludeList); const getKeywordsForCurrentSite = () => { const siteKeywords = keywords .filter(k => k.startsWith(`${window.location.hostname}:`)) .map(k => k.split(':').slice(1).join(':')); return siteKeywords.length > 0 ? siteKeywords : DEFAULT_KEYWORDS; }; const getRegexKeywords = () => { const rawKeywords = getKeywordsForCurrentSite(); return rawKeywords.map(k => { try { let pattern = k.replace(/[.*+?^${}()|[\]\\]/g, '\\$&') .replace(/\\ /g, '[\\s\\n\\r]*') .replace(/ /g, '[\\s\\n\\r]*') .replace(/\\\)\\\{/g, '\\s*\\)\\s*\\{') .replace(/\\\)\\\;/g, '\\s*\\)\\s*\\;'); return new RegExp(pattern); } catch (e) { return { test: () => false }; } }); }; const manageSite = operation => { const site = window.location.hostname; if (operation === 'exclude' && !excludeList.includes(site)) { excludeList.push(site); saveExcludeList(); } else if (operation === 'add' && excludeList.includes(site)) { excludeList = excludeList.filter(s => s !== site); saveExcludeList(); } }; const updateCount = (n) => { totalRemoved += n; GM_setValue(TOTAL_REMOVED_KEY, totalRemoved); }; const formatCount = (num) => { if (num >= 1000) return (num / 1000).toFixed(1) + 'k'; return num; }; const formatAndHighlight = (code, lang) => { if (!code) return ""; let fmt = code.replace(/\{/g, ' {\n ').replace(/\}/g, '\n}\n').replace(/;/g, ';\n ').replace(/\n\s*\n/g, '\n'); let level = 0; let result = fmt.split('\n').map(line => { line = line.trim(); if (line.includes('}')) level--; let l = ' '.repeat(Math.max(0, level)) + line; if (line.includes('{')) level++; return l; }).join('\n'); if (result.length > 8000) result = result.substring(0, 8000) + "\n...[此处脚本块过长已截断]"; const escapeHTML = (str) => str.replace(/[&<>"']/g, m => ({'&':'&','<':'<','>':'>','"':'"',"'":"'"}[m])); if (lang === 'js') { let safeJS = escapeHTML(result); return safeJS .replace(/(".*?"|'.*?'|`.*?`)/g, '$1') .replace(/\b(var|let|const|function|if|else|return|for|while|new|try|catch|async|await|case|switch|break|default)\b/g, '$1'); } else { let safeCSS = escapeHTML(result); return safeCSS .replace(/^(\s*)([^{}\n]+)(\s*\{)/gm, '$1$2$3') .replace(/:(\s*)([^;#}\n]+)(;|\n)/g, ':$1$2$3'); } }; const showInlineScripts = () => { ensureShadow(); const oldMask = shadowRoot.querySelector('.mask'); if (oldMask) oldMask.remove(); const scripts = Array.from(document.querySelectorAll('script')) .filter(s => s.innerHTML.trim()) .map((s, i) => `
${formatAndHighlight(s.innerHTML.trim(), 'js')}${formatAndHighlight(item.content, 'js')}