// ==UserScript== // @name 移动端限制弹窗解除 // @namespace https://viayoo.com/hrgow0 // @version 1.65 // @description 自用脚本,解除滑动禁止、触控禁止,移除弹窗。 // @author Via // @match *://*/* // @exclude *://joiplay.net/* // @icon  // @grant GM_addStyle // @run-at document-start // @license MIT // ==/UserScript== (function() { 'use strict'; let cleanupCount = 0, removeSelectors = [], clickSelectors = [], stayRules = new Set(), userAttemptedScroll = false, scrollRestorationAttempted = false; const MAX_CLEANUP_COUNT = 10; let alertHistory = []; const originalAlert = window.alert; const MAX_ALERTS_PER_SECOND = 3; // 弹窗和元素隐藏规则 // ##+js(dpopup, 元素名称,是否持续监控)。 // 例如 ##+js(dpopup, .ad),点击一次包含.ad的按钮,然后隐藏。 // ##+js(dpopup, .ad,stay),就是点击后不隐藏,持续监控点击多个.ad。 // 其他##就是普通的Adblock规则 // bing.com###ads_popup就是隐藏#ads_popup元素 // ###ads_popup就是通用规则 // *bing.com###ads_popup,就是模糊匹配,例如www.bing.com或者cn.bing.com都会被匹配 const Adblock_Rules = ` ###ads_popup ##.ad-dialog ##.ads-popup ##.g-pop ##.mobile-download-popup ##.modal-mask ##.overlay-mask ##.popup-mask ##.popup-overlay ##.q-dialog ##.top-advert ##.van-overlay ##[data-role="overlay"] ##+js(dpopup, .ad-area-close, stay) ##+js(dpopup, .btn-age-confirm, stay) ##+js(dpopup, .gg-close) ##+js(dpopup, .js-adv-close) ##+js(dpopup, .popup-close, stay) ##+js(dpopup, [onclick^="closePopup"], stay) bing.com##+js(dpopup, #bnp_btn_reject, stay) bing.com##+js(dpopup, #sacs_close, stay) sina.cn##+js(dpopup, #SFA-continue-btn, stay) sina.cn##+js(dpopup, .confirm-cancel) sina.cn##+js(dpopup, .snp-btn-close-new, stay) sina.cn##+js(dpopup, .unfold-mask) `.trim(); const log = (msg) => console.log(`%c[弹窗解除] ${msg}`, 'color:#1abc9c;font-weight:bold;background:#000;border-radius:4px;padding:2px 6px;'); window.alert = function(...args) { const now = Date.now(); alertHistory = alertHistory.filter(time => now - time < 1000); alertHistory.push(now); if (alertHistory.length > MAX_ALERTS_PER_SECOND) { if (window.alert === originalAlert) { window.alert = function() {}; log('alert弹窗已被禁用(1秒内超过3次)'); setTimeout(() => { window.alert = originalAlert; alertHistory = []; log('alert弹窗限制已解除'); }, 30000); } return; } return originalAlert.apply(this, args); }; function parseAdblockRules() { const currentHost = window.location.hostname; Adblock_Rules.split('\n').filter(line => line.trim()).forEach(line => { if (line.includes('##+js(dpopup,')) { const match = line.match(/##\+js\(dpopup,\s*([^,)]+)(?:,\s*(stay))?\)/); if (match) { clickSelectors.push(match[1].replace(/['"]/g, '').trim()); if (match[2] === 'stay') stayRules.add(match[1]); } } else if (line.startsWith('##')) { removeSelectors.push(line.replace(/^#{2,3}/, '').trim()); } else if (line.includes('##')) { const [domains, rule] = line.split('##'), domainList = domains.split(','); const domainMatch = domainList.some(domain => { const cleanDomain = domain.trim().replace(/^[*]+\.?/, ''); return !cleanDomain || currentHost === cleanDomain || currentHost.endsWith('.' + cleanDomain) || (cleanDomain.startsWith('*') && currentHost.endsWith(cleanDomain.slice(1))); }); if (domainMatch) { if (rule.includes('+js(dpopup,')) { const match = rule.match(/\+js\(dpopup,\s*([^,)]+)(?:,\s*(stay))?\)/); if (match) { clickSelectors.push(match[1].replace(/['"]/g, '').trim()); if (match[2] === 'stay') stayRules.add(match[1]); } } else { removeSelectors.push(rule.trim()); } } } }); removeSelectors = [...new Set(removeSelectors)]; clickSelectors = [...new Set(clickSelectors)]; } function addScrollStyles() { const css = `html,body{-webkit-overflow-scrolling:touch!important}body.no-scroll,body.modal-open,body.scroll-locked{overflow:auto!important;position:relative!important}.modal-backdrop,.popup-backdrop,.overlay-mask{display:none!important;}`; if (typeof GM_addStyle !== 'undefined') GM_addStyle(css); else { const style = document.createElement('style'); style.textContent = css; (document.head || document.documentElement).appendChild(style); } } function removePopups() { removeSelectors.forEach(selector => { try { const elements = document.querySelectorAll(selector); let hit = 0; elements.forEach(element => { if (element.offsetParent !== null) { element.style.display = 'none'; element.style.visibility = 'hidden'; element.style.opacity = '0'; if (typeof element.remove === 'function') { element.remove(); } hit++; } }); if (hit > 0) { log(`当前网页: ${location.href}\n运行规则: ${selector} ,原因: 匹配到弹窗/遮罩元素 ,状态: 成功 (已移除 ${hit} 个)`); } } catch(e) {} }); } function clickCloseButtons() { clickSelectors.forEach(selector => { try { const buttons = document.querySelectorAll(selector); let hit = 0; buttons.forEach(button => { if (button.offsetParent !== null && button.getBoundingClientRect().width > 0 && button.getBoundingClientRect().height > 0) { button.click(); hit++; if (!stayRules.has(selector)) { button.style.display = 'none'; button.style.visibility = 'hidden'; } } }); if (hit > 0) { log(`当前网页: ${location.href}\n运行规则: ${selector} ,原因: 模拟点击关闭按钮 ,状态: 成功`); } } catch(e) {} }); } function initStayObserver() { if (stayRules.size === 0) return; const observer = new MutationObserver(mutations => { stayRules.forEach(selector => { document.querySelectorAll(selector).forEach(button => { if (button.offsetParent !== null && button.getBoundingClientRect().width > 0 && button.getBoundingClientRect().height > 0) { button.click(); log(`当前网页: ${location.href}\n[stay监控] 运行规则: ${selector} ,原因: 新出现关闭按钮自动点击 ,状态: 成功`); } }); }); }); observer.observe(document.body, { childList: true, subtree: true, attributes: true }); } function isScrollAbnormal() { const bodyStyle = window.getComputedStyle(document.body); const htmlStyle = window.getComputedStyle(document.documentElement); const hasLockClass = document.body.classList.contains('no-scroll') || document.body.classList.contains('scroll-locked') || document.body.classList.contains('modal-open') || document.body.classList.contains('overflow-hidden'); const hasLockStyle = bodyStyle.overflow === 'hidden' || bodyStyle.position === 'fixed' || bodyStyle.height === '100%' && bodyStyle.overflow === 'hidden'; const contentTallerThanViewport = document.documentElement.scrollHeight > window.innerHeight; return (hasLockClass || hasLockStyle) && contentTallerThanViewport; } function enableTouchAndScroll() { ['touchmove', 'wheel', 'scroll', 'mousewheel'].forEach(eventType => { document.addEventListener(eventType, e => { if (e.defaultPrevented && e.cancelable) { e.stopImmediatePropagation(); } }, { capture: true, passive: false }); }); } document.addEventListener('wheel', function(e) { userAttemptedScroll = true; }, { passive: true }); document.addEventListener('touchmove', function(e) { userAttemptedScroll = true; }, { passive: true }); function restoreBodyScroll() { if (this === window) { scrollRestorationAttempted = false; } if (!this && scrollRestorationAttempted) return; if (!this && !userAttemptedScroll && !isScrollAbnormal()) return; let changed = false; document.body.style.cssText += 'overflow:auto!important;position:relative!important;-webkit-overflow-scrolling:touch!important;'; const style = document.body.style; if (window.getComputedStyle(document.body).overflow === 'hidden') { style.overflow = 'auto'; changed = true; } if (window.getComputedStyle(document.body).position === 'fixed') { style.position = 'relative'; style.top = ''; style.height = 'auto'; changed = true; } if (this === window) { document.documentElement.style.overflow = 'auto'; document.documentElement.style.position = 'relative'; window.scrollBy(0, 1); window.scrollBy(0, -1); } if (changed || this === window) { scrollRestorationAttempted = true; log(`手势触发恢复: ${this === window ? '用户手动双击强制解锁' : '自动修复滚动限制'} - ${location.href}`); } } function cleanup() { if (cleanupCount >= MAX_CLEANUP_COUNT) return; try { removePopups(); clickCloseButtons(); restoreBodyScroll(); cleanupCount++; } catch (error) {} } function reInject() { log(`手动触发脚本(双击页面)…… 当前网页: ${location.href}`); enableTouchAndScroll(); addScrollStyles(); removePopups(); clickCloseButtons(); restoreBodyScroll.call(window); } function initObserver() { const observer = new MutationObserver(mutations => { cleanup(); }); observer.observe(document.body, { childList: true, subtree: true, attributes: false }); } function init() { enableTouchAndScroll(); parseAdblockRules(); addScrollStyles(); setTimeout(() => { cleanup(); initObserver(); initStayObserver(); }, 500); setInterval(cleanup, 1500); document.addEventListener('dblclick', reInject); } if (document.readyState === 'loading') document.addEventListener('DOMContentLoaded', init); else setTimeout(init, 100); })();