// ==UserScript== // @name 4khd 广告屏蔽 // @namespace https://viayoo.com // @version 1.6 // @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,iframe[src*="pemsrv"],iframe[src*="magsrv"],iframe[src*="exoclick"]'; const BLOCK_REGEX = /magsrv|pemsrv|ad-provider\.js|disabley|ad-provider|exoclick|ads?[0-9]*\.|popunder|venor|popup|fxuuid|jduuid|linkSens|uuid|splash\.php/i; const PROTECT_REGEX = /popMagic|pemsrv|splash\.php|exoJsPop|BetterJsPop|disable-devtool|DisableDevtool|exoclick|magsrv/i; const currentHostname = location.hostname; const fuzzyDomain = currentHostname.replace(/\d+/g, '').replace(/\.+$/, ''); function initStorageLock() { const PROTECT_KEYS = new Set(['inData', 'inData2', 'requestClosed', 'ccc-my_favorite_post']); const COOKIE_BLOCK_RE = /fxuuid|jduuid|adsie/i; const WIN_PROPS = ['DisableDevtool', 'AdProvider', 'adConfig', 'popMagic', 'exoJsPop101', 'exoclick', 'exoJsPop']; const hostname = location.hostname; const expired = "=; expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/"; ['fxuuid', 'jduuid', 'adsie'].forEach(name => { doc.cookie = name + expired; doc.cookie = name + expired + "; domain=" + hostname; doc.cookie = name + expired + "; domain=." + hostname; }); const orgGet = Storage.prototype.getItem; const orgSet = Storage.prototype.setItem; Storage.prototype.getItem = function(key) { if (key === 'inData') return 'true'; if (key === 'inData2') return '0'; return orgGet.apply(this, arguments); }; Storage.prototype.setItem = function(key, value) { if (PROTECT_KEYS.has(key)) return; return orgSet.apply(this, arguments); }; const cookieDesc = Object.getOwnPropertyDescriptor(Document.prototype, 'cookie') || Object.getOwnPropertyDescriptor(HTMLDocument.prototype, 'cookie'); if (cookieDesc?.configurable) { Object.defineProperty(doc, 'cookie', { get: () => cookieDesc.get.call(doc), set: (val) => { if (COOKIE_BLOCK_RE.test(val)) return; cookieDesc.set.call(doc, val); }, configurable: true }); } WIN_PROPS.forEach(p => { Object.defineProperty(win, p, { value: null, writable: false, configurable: false }); }); } function initAdBlocker() { const css = `${AD_SELECTORS}{display:none!important;visibility:hidden!important;width:0!important;height:0!important;opacity:0!important;pointer-events:none!important;}`; if (hasGM) GM_addStyle(css); const randomClass = 'via-clean-' + Math.random().toString(36).substring(2, 10); const enhancedCss = `${AD_SELECTORS}{display:none!important;visibility:hidden!important;width:0!important;height:0!important;opacity:0!important;pointer-events:none!important;} .${randomClass}{display:none!important;}`; function injectStyle(useRandom = false) { const selector = useRandom ? `.${randomClass}` : 'style[data-via-adclean]'; if (document.querySelector(selector)) return; const style = document.createElement('style'); if (useRandom) style.className = randomClass; else style.dataset.viaAdclean = 'true'; style.textContent = useRandom ? enhancedCss : css; (document.head || document.documentElement || document.body || document).appendChild(style); } function earlyInject() { injectStyle(false); injectStyle(true); } if (document.documentElement) { earlyInject(); } else { document.addEventListener('readystatechange', () => { if (document.readyState !== 'loading') earlyInject(); }, { once: true }); document.addEventListener('DOMContentLoaded', earlyInject, { once: true }); } setInterval(() => { if (!document.querySelector('style[data-via-adclean]') && !document.querySelector(`.${randomClass}`)) { earlyInject(); } }, 600 + Math.random() * 400); function aggressiveClean() { const ads = document.querySelectorAll(AD_SELECTORS); if (ads.length === 0) return; ads.forEach(el => { el.remove(); }); } setTimeout(aggressiveClean, 80); setInterval(aggressiveClean, 800); } function checkExternal(href) { try { const url = new URL(href, location.href); return !url.hostname.includes('4khd') && !url.hostname.includes(fuzzyDomain); } catch (e) { return false; } } function initEventListeners() { const blockExternalLinks = (e) => { if (!e.isTrusted) return; const a = e.target.closest('a, area'); if (a?.href && checkExternal(a.href) && (a.target === '_blank' || e.ctrlKey || e.metaKey)) { e.preventDefault(); e.stopImmediatePropagation(); } }; doc.addEventListener('click', blockExternalLinks, true); } function initNetworkInterceptors() { win.open = new Proxy(win.open, { apply: (t, th, args) => (args[0] && (BLOCK_REGEX.test(args[0]) || checkExternal(args[0]))) ? null : Reflect.apply(t, th, args) }); win.fetch = new Proxy(win.fetch, { apply: (t, th, args) => { const url = args[0]?.url || args[0]; if (typeof url === 'string' && BLOCK_REGEX.test(url)) { return Promise.resolve(new Response('', { status: 200, statusText: 'OK' })); } return Reflect.apply(t, th, args); } }); const orgOpen = XMLHttpRequest.prototype.open; XMLHttpRequest.prototype.open = function(_, url) { if (typeof url === 'string' && BLOCK_REGEX.test(url)) { this._blocked = true; return; } return orgOpen.apply(this, arguments); }; const orgSend = XMLHttpRequest.prototype.send; XMLHttpRequest.prototype.send = function() { if (this._blocked) { this.dispatchEvent(new Event('load')); this.dispatchEvent(new Event('readystatechange')); return; } return orgSend.apply(this, arguments); }; } function initMutationObserver() { const observer = new MutationObserver(muts => { muts.forEach(mut => { mut.addedNodes.forEach(node => { if (node.nodeType !== 1) return; if (node.tagName === 'SCRIPT' && PROTECT_REGEX.test(node.textContent + node.src)) { node.remove(); return; } if (node.matches?.(AD_SELECTORS)) { node.remove(); } else { node.querySelectorAll?.(AD_SELECTORS).forEach(el => el.remove()); } }); }); }); observer.observe(doc.documentElement, { childList: true, subtree: true }); } function addImageButtons() { const style = doc.createElement('style'); style.textContent = ` .img-button-container{position:absolute;bottom:15px;right:10px;display:flex;gap:8px;z-index:100} .img-btn{padding:8px 16px;border:1px solid rgba(255,255,255,0.25);border-radius:16px;cursor:pointer;font-size:13px;font-weight:600;backdrop-filter:blur(16px);background:rgba(255,255,255,0.3);color:#1C2526;transition:all 0.2s;font-family:sans-serif} @media (prefers-color-scheme:dark){.img-btn{background:rgba(40,40,40,0.3);color:#f0f0f0}} .img-btn:hover{background:rgba(255,255,255,0.4);transform:translateY(-1px)} .image-wrapper{position:relative;display:inline-block} `; doc.head.appendChild(style); const processImages = () => { doc.querySelectorAll('#basicExample > a.imageLink').forEach(link => { if (link.querySelector('.img-button-container')) return; const wrapper = doc.createElement('div'); wrapper.className = 'image-wrapper'; link.parentNode.insertBefore(wrapper, link); wrapper.appendChild(link); const container = doc.createElement('div'); container.className = 'img-button-container'; const btn = doc.createElement('button'); btn.className = 'img-btn'; btn.textContent = '打开'; const img = link.querySelector('img'); const url = img?.src || img?.dataset?.src || link.href; btn.onclick = (e) => { e.preventDefault(); e.stopPropagation(); const a = doc.createElement('a'); a.href = url; a.download = url.split('/').pop(); a.click(); }; container.appendChild(btn); wrapper.appendChild(container); }); }; const imageObserver = new MutationObserver(processImages); imageObserver.observe(doc.body || doc.documentElement, { childList: true, subtree: true }); processImages(); } function initAll() { initAdBlocker(); initStorageLock(); initEventListeners(); initNetworkInterceptors(); initMutationObserver(); addImageButtons(); } if (doc.readyState === 'loading') { doc.addEventListener('DOMContentLoaded', initAll); } else { initAll(); } })();