// ==UserScript== // @name 4khd 广告屏蔽 // @namespace https://viayoo.com // @version 1.2 // @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 keys = ['inData', 'inData2', 'requestClosed', 'ccc-my_favorite_post']; const orgGet = Storage.prototype.getItem; Storage.prototype.getItem = function(key) { if (key === 'inData') return 'true'; return orgGet.apply(this, arguments); }; const orgSet = Storage.prototype.setItem; Storage.prototype.setItem = function(key, value) { if (keys.includes(key)) return; return orgSet.apply(this, arguments); }; const orgCookie = doc.cookie; Object.defineProperty(doc, 'cookie', { get: () => orgCookie, set: (val) => { if (val.includes('fxuuid')) return; return val; }, configurable: true }); } 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 initStaticBlock() { const css = `${AD_SELECTORS}{display:none!important;visibility:hidden!important;pointer-events:none!important;z-index:-999!important;height:0!important;}`; if (hasGM) { GM_addStyle(css); } else { const s = doc.createElement('style'); s.textContent = css; (doc.head || doc.documentElement).appendChild(s); } ['DisableDevtool', 'AdProvider', 'adConfig', 'popMagic', 'exoJsPop101', 'exoclick', 'exoJsPop'].forEach(p => { Object.defineProperty(win, p, { value: null, writable: false, configurable: false }); }); } 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]; return (typeof url === 'string' && BLOCK_REGEX.test(url)) ? Promise.resolve(new Response('', { status: 200, statusText: 'OK', headers: { 'Content-Type': 'text/plain' } })) : Reflect.apply(t, th, args); } }); const XHR = XMLHttpRequest.prototype; const _open = XHR.open; const _send = XHR.send; XHR.open = function(_, url) { if (typeof url === 'string' && BLOCK_REGEX.test(url)) { this._blocked = true; Object.defineProperty(this, 'status', { value: 200 }); Object.defineProperty(this, 'readyState', { value: 4 }); Object.defineProperty(this, 'responseText', { value: '' }); } return this._blocked ? null : _open.apply(this, arguments); }; XHR.send = function() { if (this._blocked) { if (typeof this.onreadystatechange === 'function') { this.onreadystatechange(); } return; } return _send.apply(this, arguments); }; } function createImageBtn(url) { const container = doc.createElement('div'); container.className = 'img-button-container'; const btn = doc.createElement('button'); btn.className = 'img-btn'; btn.textContent = '打开'; 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); return container; } function processImages(root = doc) { const links = root.querySelectorAll('#basicExample > a.imageLink'); links.forEach(link => { if (link.dataset.processed || link.parentNode.classList.contains('image-wrapper')) return; link.dataset.processed = "true"; const wrapper = doc.createElement('div'); wrapper.className = 'image-wrapper'; link.parentNode.insertBefore(wrapper, link); wrapper.appendChild(link); const img = link.querySelector('img'); const targetUrl = img?.src || img?.dataset?.src || link.href; wrapper.appendChild(createImageBtn(targetUrl)); }); } function initUnifiedObserver() { const style = doc.createElement('style'); style.textContent = ` .image-wrapper{position:relative;display:inline-block} .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} @media (prefers-color-scheme:dark){.img-btn{background:rgba(40,40,40,0.3);color:#f0f0f0}} .img-btn:hover{transform:translateY(-1px);background:rgba(255,255,255,0.4)} `; doc.head.appendChild(style); const observer = new MutationObserver(mutations => { for (const mutation of mutations) { for (const node of mutation.addedNodes) { if (node.nodeType !== 1) continue; if (node.tagName === 'SCRIPT' && PROTECT_REGEX.test(node.textContent || node.src || '')) { node.remove(); continue; } if (node.matches?.(AD_SELECTORS)) { node.remove(); } processImages(node); } } }); observer.observe(doc.documentElement, { childList: true, subtree: true }); processImages(); } function initAll() { initStorageLock(); initStaticBlock(); initNetworkInterceptors(); initUnifiedObserver(); doc.addEventListener('click', (e) => { if (!e.isTrusted) return; const a = e.target.closest('a, area'); if (a?.href && checkExternal(a.href) && (a.target === '_blank' || e.ctrlKey)) { e.preventDefault(); e.stopImmediatePropagation(); } }, true); setInterval(() => { doc.querySelectorAll(AD_SELECTORS).forEach(el => el.remove()); processImages(); }, 2000); } if (doc.readyState === 'loading') { doc.addEventListener('DOMContentLoaded', initAll); } else { initAll(); } })();