// ==UserScript== // @name 4khd 广告屏蔽 // @namespace https://viayoo.com // @version 0.51 // @description 移除4khd广告,兼容原生和GM环境。 // @author Via // @license MIT // @match *://*.4khd.com/* // @match *://*.xxtt.ink/* // @match *://*.uuss.uk/* // @match *://*.ssuu.uk/* // @icon  // @run-at document-start // @grant unsafeWindow // @grant GM_addStyle // ==/UserScript== (function() { 'use strict'; const hasGM = typeof GM_addStyle === 'function' && typeof unsafeWindow !== 'undefined'; const win = hasGM ? unsafeWindow : window; const doc = document; const AD_SELECTORS = '.exo_wrapper,.popup,.centbtd,.exo-native-widget,.exo-native-widget-outer-container,ins[data-processed="true"],.popup-iframe,ins.adsbynetwork,.wb-contai'; const BLOCK_PATTERNS = [/magsrv|pemsrv|ad-provider\.js|disabley|ad-provider|exoclick|ads?[0-9]*\.|popunder|venor|popup|fxuuid|jduuid|linkSens|uuid|splash\.php/i]; let lastDomain = location.hostname; const cleanStorage = () => { ['storedResult','inData','extranks','Better','linkSens','jduuid','adshsi','inData2','BetterJsPop_lastOpenedAt'].forEach(k => { localStorage.removeItem(k); sessionStorage.removeItem(k); }); ['adsie','jduuid'].forEach(n => { document.cookie = `${n}=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/`; document.cookie = `${n}=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/; domain=.${location.hostname}`; document.cookie = `${n}=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/; domain=${location.hostname}`; }); }; const checkDomainChange = () => { const cur = location.hostname; if (cur !== lastDomain) { cleanStorage(); lastDomain = cur; } }; if (hasGM) { GM_addStyle(`${AD_SELECTORS}{display:none!important;visibility:hidden!important}`); } else { const s = doc.createElement('style'); s.textContent = `${AD_SELECTORS}{display:none!important;visibility:hidden!important}`; (doc.head || doc.documentElement).appendChild(s); } ['AdProvider','adConfig','popMagic','exoJsPop101','exoclick','exoJsPop'].forEach(p => { Object.defineProperty(win, p, {value: null, writable: false, configurable: false}); }); const shouldBlock = url => url && BLOCK_PATTERNS.some(p => p.test(url)); const blockDangerousClick = e => { if (!e.isTrusted) return; const t = e.target; if (t.closest('a, button, [role="button"], [onclick]') && (shouldBlock(t.href) || shouldBlock(t.getAttribute('data-url')))) { e.preventDefault(); e.stopPropagation(); e.stopImmediatePropagation(); return false; } }; doc.addEventListener('click', blockDangerousClick, true); doc.addEventListener('mousedown', blockDangerousClick, true); window.open = new Proxy(window.open, { apply: (t, th, args) => shouldBlock(args[0]) ? null : t.apply(th, args) }); window.fetch = new Proxy(window.fetch, { apply: (t, th, args) => shouldBlock(args[0]?.url || args[0]) ? Promise.resolve(new Response('', {status: 204})) : t.apply(th, args) }); const originalXHRopen = XMLHttpRequest.prototype.open; XMLHttpRequest.prototype.open = function() { if (shouldBlock(arguments[1])) { this._blocked = true; return; } return originalXHRopen.apply(this, arguments); }; const originalXHRSend = XMLHttpRequest.prototype.send; XMLHttpRequest.prototype.send = function() { return this._blocked ? undefined : originalXHRSend.apply(this, arguments); }; const originalAddEvent = EventTarget.prototype.addEventListener; EventTarget.prototype.addEventListener = function(type, listener, options) { if (type === 'click' && typeof listener === 'function') { const str = listener.toString(); if (/popMagic|pemsrv|splash\.php|exoJsPop|BetterJsPop/.test(str)) return; } return originalAddEvent.call(this, type, listener, options); }; doc.querySelector = new Proxy(doc.querySelector, { apply(target, thisArg, args) { return args[0] === "[disable-devtool-auto]" ? null : target.apply(thisArg, args); } }); const originalEval = win.eval; win.eval = code => { if (typeof code === 'string' && /disable-devtool|DisableDevtool/i.test(code)) return; return originalEval.call(win, code); }; const originalFunction = win.Function; win.Function = (...args) => { const body = args[args.length - 1]; if (typeof body === 'string' && /disable-devtool|DisableDevtool/i.test(body)) return () => {}; return originalFunction(...args); }; Object.defineProperty(win, 'debugger', { get: () => function() {}, set: () => {}, configurable: false, enumerable: false }); const observer = new MutationObserver(muts => { muts.forEach(mut => { mut.addedNodes.forEach(node => { if (node.nodeType !== 1) return; if (node.tagName === 'SCRIPT') { const txt = node.textContent || ''; const src = node.src || ''; if (txt.includes('popMagic') || txt.includes('splash.php') || txt.includes('BetterJsPop') || /disable-devtool|DisableDevtool|pemsrv|magsrv|exoclick/i.test(txt + src)) { node.remove(); return; } } if (node.tagName === 'IFRAME' && (shouldBlock(node.src) || node.src === 'about:blank')) { node.remove(); } }); }); }); observer.observe(doc.documentElement, {childList: true, subtree: true}); const aggressiveClean = () => { doc.querySelectorAll(AD_SELECTORS).forEach(el => el.remove()); doc.querySelectorAll('iframe[src*="pemsrv"],iframe[src*="magsrv"],iframe[src*="exoclick"]').forEach(el => el.remove()); cleanStorage(); }; setInterval(aggressiveClean, 500); if (doc.readyState === 'loading') { doc.addEventListener('DOMContentLoaded', aggressiveClean); } else { aggressiveClean(); } const originalPushState = history.pushState; const originalReplaceState = history.replaceState; history.pushState = function() { const result = originalPushState.apply(this, arguments); setTimeout(checkDomainChange, 0); return result; }; history.replaceState = function() { const result = originalReplaceState.apply(this, arguments); setTimeout(checkDomainChange, 0); return result; }; })();