// ==UserScript==
// @name YouTube去广告 YouTube AD Blocker
// @name:zh-CN YouTube去广告
// @name:zh-TW YouTube去廣告
// @name:zh-HK YouTube去廣告
// @name:zh-MO YouTube去廣告
// @namespace http://tampermonkey.net/
// @version 5.97
// @description 这是一个去除YouTube广告的脚本,轻量且高效,它能丝滑的去除界面广告和视频广告,包括6s广告。This is a script that removes ads on YouTube, it's lightweight and efficient, capable of smoothly removing interface and video ads, including 6s ads.
// @description:zh-CN 这是一个去除YouTube广告的脚本,轻量且高效,它能丝滑的去除界面广告和视频广告,包括6s广告。
// @description:zh-TW 這是一個去除YouTube廣告的腳本,輕量且高效,它能絲滑地去除界面廣告和視頻廣告,包括6s廣告。
// @description:zh-HK 這是一個去除YouTube廣告的腳本,輕量且高效,它能絲滑地去除界面廣告和視頻廣告,包括6s廣告。
// @description:zh-MO 這是一個去除YouTube廣告的腳本,輕量且高效,它能絲滑地去除界面廣告和視頻廣告,包括6s廣告。
// @author iamfugui
// @match *://www.youtube.com/*
// @match *://m.youtube.com/*
// @icon https://www.google.com/s2/favicons?sz=64&domain=YouTube.com
// @grant none
// @license MIT
// ==/UserScript==
(function() {
`use strict`;
//界面广告选择器
const cssSeletorArr = [
`#masthead-ad`,//首页顶部横幅广告.
`ytd-rich-item-renderer.style-scope.ytd-rich-grid-row #content:has(.ytd-display-ad-renderer)`,//首页视频排版广告.
`.video-ads.ytp-ad-module`,//播放器底部广告.
`tp-yt-paper-dialog:has(yt-mealbar-promo-renderer)`,//播放页会员促销广告.
`ytd-engagement-panel-section-list-renderer[target-id="engagement-panel-ads"]`,//播放页右上方推荐广告.
`#related #player-ads`,//播放页评论区右侧推广广告.
`#related ytd-ad-slot-renderer`,//播放页评论区右侧视频排版广告.
`ytd-ad-slot-renderer`,//搜索页广告.
`yt-mealbar-promo-renderer`,//播放页会员推荐广告.
`ad-slot-renderer`,//M播放页第三方推荐广告
`ytm-companion-ad-renderer`,//M可跳过的视频广告链接处
];
window.dev=false;//开发使用
/**
* 将标准时间格式化
* @param {Date} time 标准时间
* @param {String} format 格式
* @return {String}
*/
function moment(time, format = `YYYY-MM-DD HH:mm:ss`) {
// 获取年⽉⽇时分秒
let y = time.getFullYear()
let m = (time.getMonth() + 1).toString().padStart(2, `0`)
let d = time.getDate().toString().padStart(2, `0`)
let h = time.getHours().toString().padStart(2, `0`)
let min = time.getMinutes().toString().padStart(2, `0`)
let s = time.getSeconds().toString().padStart(2, `0`)
if (format === `YYYY-MM-DD`) {
return `${y}-${m}-${d}`
} else {
return `${y}-${m}-${d} ${h}:${min}:${s}`
}
}
/**
* 输出信息
* @param {String} msg 信息
* @return {undefined}
*/
function log(msg) {
if(!window.dev){
return false;
}
console.log(`${moment(new Date())} ${msg}`)
}
/**
* 设置运行标志
* @param {String} name
* @return {undefined}
*/
function setRunFlag(name){
let style = document.createElement(`style`);
style.id = name;
(document.querySelector(`head`) || document.querySelector(`body`)).appendChild(style);//将节点附加到HTML.
}
/**
* 获取运行标志
* @param {String} name
* @return {undefined|Element}
*/
function getRunFlag(name){
return document.getElementById(name);
}
/**
* 检查是否设置了运行标志
* @param {String} name
* @return {Boolean}
*/
function checkRunFlag(name){
if(getRunFlag(name)){
return true;
}else{
setRunFlag(name)
return false;
}
}
/**
* 生成去除广告的css元素style并附加到HTML节点上
* @param {String} styles 样式文本
* @return {undefined}
*/
function generateRemoveADHTMLElement(styles) {
//如果已经设置过,退出.
if (checkRunFlag(`RemoveADHTMLElement`)) {
log(`屏蔽页面广告节点已生成`);
return false
}
//设置移除广告样式.
let style = document.createElement(`style`);//创建style元素.
(document.querySelector(`head`) || document.querySelector(`body`)).appendChild(style);//将节点附加到HTML.
style.appendChild(document.createTextNode(styles));//附加样式节点到元素节点.
log(`生成屏蔽页面广告节点成功`)
}
/**
* 生成去除广告的css文本
* @param {Array} cssSeletorArr 待设置css选择器数组
* @return {String}
*/
function generateRemoveADCssText(cssSeletorArr){
cssSeletorArr.forEach((seletor,index)=>{
cssSeletorArr[index]=`${seletor}{display:none!important}`;//遍历并设置样式.
});
return cssSeletorArr.join(` `);//拼接成字符串.
}
/**
* 触摸事件
* @return {undefined}
*/
function nativeTouch(){
const minNum = 375;
const maxNum = 750;
const randomNum = (Math.floor(Math.random() * (maxNum - minNum + 1)) + minNum)/1000;
let element =this;
// 创建 Touch 对象
let touch = new Touch({
identifier: Date.now(),
target: element,
clientX: 12+randomNum,
clientY: 34+randomNum,
radiusX: 56+randomNum,
radiusY: 78+randomNum,
rotationAngle: 0,
force: 1
});
// 创建 TouchEvent 对象
let touchStartEvent = new TouchEvent("touchstart", {
bubbles: true,
cancelable: true,
view: window,
touches: [touch],
targetTouches: [touch],
changedTouches: [touch]
});
// 分派 touchstart 事件到目标元素
element.dispatchEvent(touchStartEvent);
// 创建 TouchEvent 对象
let touchEndEvent = new TouchEvent("touchend", {
bubbles: true,
cancelable: true,
view: window,
touches: [],
targetTouches: [],
changedTouches: [touch]
});
// 分派 touchend 事件到目标元素
element.dispatchEvent(touchEndEvent);
}
/**
* 跳过广告
* @return {undefined}
*/
function skipAd(mutationsList, observer) {
let video = document.querySelector(`.ad-showing video`) || document.querySelector(`video`);//获取视频节点
let skipButton = document.querySelector(`.ytp-ad-skip-button`) || document.querySelector(`.ytp-ad-skip-button-modern`);
let shortAdMsg = document.querySelector(`.video-ads.ytp-ad-module .ytp-ad-player-overlay`);
if(skipButton){
//拥有跳过按钮的广告.
log(`总时长:`);
log(`${video.duration}`)
log(`当前时间:`);
log(`${video.currentTime}`)
// 跳过广告.
skipButton.click();//PC
nativeTouch.call(skipButton);//Phone
log(`按钮跳过了该广告~~~~~~~~~~~~~`);
}else if(shortAdMsg){
//没有跳过按钮的短广告.
log(`总时长:`);
log(`${video.duration}`)
log(`当前时间:`);
log(`${video.currentTime}`)
video.currentTime = video.duration;
log(`强制结束了该广告~~~~~~~~~~~~~`);
}else{
log(`######广告不存在######`);
}
}
/**
* 去除播放中的广告
* @return {undefined}
*/
function removePlayerAD(){
//如果已经在运行,退出.
if (checkRunFlag(`removePlayerAD`)) {
log(`去除播放中的广告功能已在运行`);
return false
}
let observer;//监听器
let timerID;//定时器
//开始监听
function startObserve(){
//广告节点监听
const targetNode = document.querySelector(`.video-ads.ytp-ad-module`);
if(!targetNode){
log(`正在寻找待监听的目标节点`);
return false;
}
//监听视频中的广告并处理
const config = {childList: true, subtree: true };// 监听目标节点本身与子树下节点的变动
observer = new MutationObserver(skipAd);// 创建一个观察器实例并设置处理广告的回调函数
observer.observe(targetNode, config);// 以上述配置开始观察广告节点
timerID=setInterval(skipAd, 1000);//漏网鱼
}
//轮询任务
setInterval(function(){
//视频播放页
if(observer){
return false;
}
startObserve();
},16);
log(`运行去除播放中的广告功能成功`)
}
/**
* main函数
*/
function main(){
generateRemoveADHTMLElement(generateRemoveADCssText(cssSeletorArr));//移除界面中的广告.
removePlayerAD();//移除播放中的广告.
}
if (document.readyState === `loading`) {
log(`YouTube去广告脚本即将调用:`);
document.addEventListener(`DOMContentLoaded`, main);// 此时加载尚未完成
} else {
log(`YouTube去广告脚本快速调用:`);
main();// 此时`DOMContentLoaded` 已经被触发
}
})();