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());
}