123读书网去广告

移除手机版的123读书网中的广告,如果有同种类型的广告的网站可以适配到该脚本中。

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @name         123读书网去广告
// @namespace    https://github.com/RANSAA
// @version      0.0.3
// @description  移除手机版的123读书网中的广告,如果有同种类型的广告的网站可以适配到该脚本中。
// @author       sayaDev
// @license      MIT License
// @icon         

// @match        *://m.123dua.com/*
// @grant        none


// @compatible   chrome
// @compatible   edge
// @compatible   firefox
// @compatible   opera 
// @compatible   safari 

// @run-at document-end


// ==/UserScript==
/**
 * 说明:
 * 
 * noframes: 禁止脚本在iframe中运行
 * 
 * compatible:greasyfork.org脚本站点兼容性图标
 * 
 * run-at document-end:在文档结束时执行,但是在页面上的资源如图片和样式表加载完成之前。在DOM完全解析后立即执行。
 * 
 **/ 



const name = '123读书网去广告:';

(function() {
    'use strict';

    removeAD();
})();



function removeAD(){
    // //移除顶部悬浮广告
    // removeStyleFlexAD()

    // //移除底部ID为“nsws”的广告DIV标签
    // removeAllDivWithID("nsws");


    removeAllDIVNodesWithSpecifiedID();

    removeAllDynamicStyleNodes();
}


//----------------------------------------------------------------------------------------remove id div----------------------------------------------------------------------------------------
/**
 * 删除所有指定id的DIV节点
 **/ 
function removeAllDIVNodesWithSpecifiedID(){
	//移除底部ID为“nsws”的广告DIV标签
	removeDIVNodeWithID("nsws");
}

/**
 * 删除指定ID的DIV节点
 **/ 

function removeDIVNodeWithID(id){
	var divID = id;
	var isRemove = false;
	var allElements = document.querySelectorAll('*');
	for (var i = allElements.length - 1; i >= 0; i--) {
	  if (allElements[i].id === divID) {
	    console.log(name+"移除id为:"+divID+"的DIV标签")
	    console.log(allElements[i])
	    allElements[i].remove();
	    isRemove = true;
	  }
	}
	if (!isRemove) {
		//console.log(name+"没有找到id为:"+id+"的DIV标签");
	}
}
//----------------------------------------------------------------------------------------remove id div----------------------------------------------------------------------------------------





//----------------------------------------------------------------------------------------remove style----------------------------------------------------------------------------------------

/**
 * 移除所有动态的style节点
 */ 
function removeAllDynamicStyleNodes(){
	//移除顶部悬浮广告
	//123读书网动态广告style内容匹配正则表达式
	const styleContentRegex1 = /body{margin: 0px;}#(\w+) ~ \.(\w+){display: block; width: 10%; position: fixed; z-index: 2147483646;}#(\w+) ~ /;
	removeStyleWithRegex(styleContentRegex1);

	//移除123读书网顶部悬浮广告  -  弃用的正则表达式
	const styleContentRegex1_Deprecation = /body{margin-top:120px;padding-top:0 !important;}#(\w+){display:flex;}/;
	removeStyleWithRegex(styleContentRegex1_Deprecation);


	const styleContentRegex2 = /#(\w+) ~ \.(\w+){background-image: url\(/;
	removeStyleWithRegex(styleContentRegex2);




	//动态节点可能是由其它JS动态生成,所以想要移除,需要多次重复执行
	//每0.2秒执行一次,共执行10次 
	let count = 0;
	const interval = setInterval(() => {
	  // 执行目标函数
		if (count > 2) {
			removeStyleWithRegex(styleContentRegex1);
			removeStyleWithRegex(styleContentRegex1_Deprecation);
			removeStyleWithRegex(styleContentRegex2);		
		}
	  


		count++;
		if (count >= 12) {
		clearInterval(interval); // 停止定时器
		}
	}, 200);

}

/**
 * 利用正则表达式动态匹配style并删除style节点
 **/ 
function removeStyleWithRegex(styleContentRegex){
	// 指定要匹配的样式内容的正则表达式,例如匹配包含 ".target-class" 的样式
	// const styleContentRegex = /body{margin: 0px;}#(\w+) ~ \.(\w+){display: block; width: 10%; position: fixed; z-index: 2147483646;}#(\w+) ~ /;

	// 获取所有 <style> 标签
	const styleTags = document.querySelectorAll('style');
	var isRemove = false;

	// 遍历每个 <style> 标签
	styleTags.forEach(styleTag => {
	  // 检查 <style> 标签的内容是否匹配正则表达式
	  if (styleContentRegex.test(styleTag.innerHTML)) {
	    // 如果匹配,则移除该 <style> 标签
	    styleTag.remove();
	    isRemove = true;
	    console.log(name+"查找到了动态AD style:")
	    console.log(styleTag)

	    // 获取并输出匹配的 <style> 标签的 id 属性
	    const styleId = styleTag.getAttribute('id');
	    console.log(name, "匹配的 <style> 标签 ID:", styleId);

	    // 获取 <style> 标签中的所有类名(形如 .zXzyCei)
	    const classRegex = /\.(\w[-\w]*)/g;
	    const classes = [...styleTag.innerHTML.matchAll(classRegex)].map(match => match[1]);
	    console.log(name, "style中匹配到的Class名称:", classes);
	    classes.forEach( className => {
	    	//console.log(className);
	    	removeAllTagWithClass(className);
	    })

	    // 获取 <style> 标签中的所有 ID 选择器(形如 #VoUnOkp)
	    const idRegex = /#(\w[-\w]*)/g;
	    const ids = [...styleTag.innerHTML.matchAll(idRegex)].map(match => match[1]);
	    console.log(name, "style中匹配到的ID名称:", ids);
	    ids.forEach( idName => {
	    	//console.log(idName);
	    	removeAllTagWithID(idName);
	    })
	  }
	});
	if (!isRemove) {
		//console.log(name+"动态AD style查找失败!");
	}
}

//----------------------------------------------------------------------------------------remove style----------------------------------------------------------------------------------------






/**
 * 移除所有targetId的标签
 */ 
function removeAllTagWithID(targetId){
	//检查输入的ID选择器是否有效
	if (!/^[a-zA-Z_]/.test(targetId)) {
		return;
	}


	// // 使用模板字符串构造选择器
	// const divs = document.querySelectorAll(`div#${targetId}`);
	// // 遍历并移除每个匹配的 <div> 标签
	// divs.forEach(div => div.remove());


	// 使用模板字符串构造选择器,选择所有具有该 ID 的元素
	const elements = document.querySelectorAll(`#${targetId}`);
	// 遍历并移除每个匹配的元素
	elements.forEach(element => element.remove());
}


/**
 * 移除所有targetClass的标签
 **/
function removeAllTagWithClass(targetClass){
	// 使用模板字符串构造选择器,选择所有具有该类名的元素
	const elements = document.querySelectorAll(`.${targetClass}`);
	// 遍历并移除每个匹配的元素
	elements.forEach(element => element.remove());
}