YouTube-adb

移除YouTube广告,包括静态广告、视频广告和Shorts广告,兼容最新YouTube页面结构。

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey, Greasemonkey или Violentmonkey.

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey или Violentmonkey.

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey или Violentmonkey.

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey или Userscripts.

За да инсталирате скрипта, трябва да инсталирате разширение като Tampermonkey.

За да инсталирате този скрипт, трябва да имате инсталиран скриптов мениджър.

(Вече имам скриптов мениджър, искам да го инсталирам!)

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

(Вече имам инсталиран мениджър на стиловете, искам да го инсталирам!)

// ==UserScript==
// @name         YouTube-adb
// @name:zh-CN   YouTube去广告
// @namespace    https://github.com/yutian81/greasyfork-js
// @version      7.0
// @description  移除YouTube广告,包括静态广告、视频广告和Shorts广告,兼容最新YouTube页面结构。
// @match        *://*.youtube.com/*
// @exclude      *://accounts.youtube.com/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=youtube.com
// @grant        none
// @license      MIT
// ==/UserScript==

(function () {
    'use strict';

    let video;
    const CSS_SELECTORS = [
        `#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`, // 播放页会员推荐广告
        `ytd-popup-container:has(a[href="/premium"])`, // 会员拦截广告
        `ad-slot-renderer`, // 移动端第三方推荐广告
        `ytm-companion-ad-renderer`, // 移动端可跳过的视频广告链接处
        `ytd-reel-player-renderer #ad-container`, // YouTube Shorts广告
    ];

    window.dev = false; // 开发模式开关

    /**
     * 格式化时间
     */
    function formatTime(date) {
        return date.toISOString().replace(/T/, ' ').replace(/\..+/, '');
    }

    /**
     * 输出日志
     */
    function log(message) {
        if (window.dev) {
            console.log(`[${formatTime(new Date())}] ${message}`);
        }
    }

    /**
     * 生成移除广告的CSS样式
     */
    function generateCSS() {
        const style = document.createElement('style');
        style.id = 'youtube-adblocker-style';
        style.textContent = CSS_SELECTORS.map(selector => `${selector}{display:none!important}`).join(' ');
        (document.head || document.body).appendChild(style);
        log('生成屏蔽广告样式成功');
    }

    /**
     * 跳过广告
     */
    function skipAd() {
        const skipButton = document.querySelector('.ytp-ad-skip-button, .ytp-skip-ad-button, .ytp-ad-skip-button-modern');
        const shortAdMsg = document.querySelector('.video-ads.ytp-ad-module .ytp-ad-player-overlay, .ytp-ad-button-icon');

        if (skipButton) {
            skipButton.click();
            log('点击跳过广告按钮');
        } else if (shortAdMsg) {
            video.currentTime = video.duration;
            log('强制结束广告');
        }
    }

    /**
     * 处理视频广告
     */
    function handleVideoAds() {
        video = document.querySelector('.ad-showing video') || document.querySelector('video');
        if (video && video.classList.contains('ad-showing')) {
            skipAd();
        }
    }

    /**
     * 监听DOM变化
     */
    function observeDOM() {
        const observer = new MutationObserver(() => {
            handleVideoAds();
        });
        observer.observe(document.body, { childList: true, subtree: true });
        log('开始监听DOM变化');
    }

    /**
     * 主函数
     */
    function main() {
        generateCSS();
        observeDOM();
        log('YouTube去广告脚本已启动');
    }

    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', main);
    } else {
        main();
    }
})();