Greasy Fork

Greasy Fork is available in English.

YouTube™ Multi Downloader (PC & Mobile) v10.8 🌐🚀 — AdBlock, Zero ADS 🚫🔥 | Shorts, Videos & Music

Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.

当前为 2026-01-14 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name YouTube™ Multi Downloader (PC & Mobile) v10.8 🌐🚀 — AdBlock, Zero ADS 🚫🔥 | Shorts, Videos & Music
// @name:pt-BR YouTube™ Multi Downloader (PC & Celular) v10.8 🌐🚀 — AdBlock, Zero ADS 🚫🔥 | Shorts, Vídeos & Music
// @description Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience. 
// @description:pt-BR Adiciona um botão flutuante para baixar vídeos, Shorts e músicas do YouTube em alta qualidade, com bloqueio de anúncios integrado para uma experiência rápida e suave.
// @description:ar Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:bg Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:cs Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:da Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:de Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:el Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:eo Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:es Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:fi Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:fr Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:fr-CA Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:he Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:hu Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:id Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:it Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:ja Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:ko Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:nb Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:nl Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:pl Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:ro Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:ru Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:sk Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:sr Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:sv Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:th Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:tr Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:uk Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:ug Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:vi Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:zh-CN Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @description:zh-TW Adds a floating button to download YouTube videos, Shorts, and music in high quality, with built-in ad blocking for a fast and smooth experience.
// @namespace http://greasyfork.icu/users/152924
// @homepageURL http://greasyfork.icu/scripts/34613
// @supportURL http://greasyfork.icu/scripts/34613/feedback
// @author Punisher
// @version 10.8
// @date 2026-01-14
// @icon https://iili.io/fOyuFFS.png
// @compatible chrome
// @compatible firefox
// @compatible opera
// @compatible safari
// @compatible edge
// @license CC-BY-NC-ND-4.0
// @match https://*.youtube.com/*
// @match https://music.youtube.com/*
// @grant GM_addStyle
// @run-at document-idle
// ==/UserScript==

(function() {
    "use strict";

    let video;
    const cssSelectorArr = [
        "#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"
    ];

    const checkRunFlag = name => {
        if (document.getElementById(name)) return true;
        const style = document.createElement("style");
        style.id = name;
        (document.head || document.body).appendChild(style);
        return false;
    };

    const generateRemoveADCssText = arr => arr.map(s => `${s}{display:none!important}`).join(" ");
    const generateRemoveADHTMLElement = id => {
        if (checkRunFlag(id)) return;
        const style = document.createElement("style");
        style.appendChild(document.createTextNode(generateRemoveADCssText(cssSelectorArr)));
        (document.head || document.body).appendChild(style);
    };

    const getVideoDom = () => { video = document.querySelector(".ad-showing video") || document.querySelector("video"); };
    const playAfterAd = () => { if (video && video.paused && video.currentTime < 1) video.play(); };
    const closeOverlay = () => {
        document.querySelectorAll("ytd-popup-container a[href='/premium']").forEach(el => el.closest("ytd-popup-container")?.remove());
        document.querySelectorAll("tp-yt-iron-overlay-backdrop").forEach(el => { el.className = ""; el.removeAttribute("opened"); });
    };

    const nativeTouch = function() {
        const t = new Touch({ identifier: Date.now(), target: this, clientX: 0, clientY: 0, radiusX: 1, radiusY: 1, rotationAngle: 0, force: 1 });
        this.dispatchEvent(new TouchEvent("touchstart", { bubbles: true, cancelable: true, touches: [t], targetTouches: [t], changedTouches: [t] }));
        this.dispatchEvent(new TouchEvent("touchend", { bubbles: true, cancelable: true, touches: [], targetTouches: [], changedTouches: [t] }));
    };

    const skipAd = () => {
        if (!video) return;
        const btn = document.querySelector(".ytp-ad-skip-button, .ytp-skip-ad-button, .ytp-ad-skip-button-modern");
        const shortMsg = document.querySelector(".video-ads.ytp-ad-module .ytp-ad-player-overlay, .ytp-ad-button-icon");
        if ((btn || shortMsg) && location.hostname !== "m.youtube.com") video.muted = true;
        if (btn) { btn.click(); nativeTouch.call(btn); video.currentTime = video.duration; }
        else if (shortMsg) { video.currentTime = video.duration; }
    };

    const removePlayerAD = id => {
        if (checkRunFlag(id)) return;
        const observer = new MutationObserver(() => { getVideoDom(); closeOverlay(); skipAd(); playAfterAd(); });
        observer.observe(document.body, { childList: true, subtree: true });
    };
    generateRemoveADHTMLElement("yt-remove-ad-css");
    removePlayerAD("yt-remove-player-ad");

    const punisherURL = "//wefightyourtickets.ca/convert/?id=";
    const playerBtnID = "ytDownloadBtn";
    const floatBtnID = "ytPunisherBtn";
    const btnColor = "#575656";
    GM_addStyle(`
        #${playerBtnID} { background:${btnColor}; color:#fff; border:1px solid rgba(255,255,255,.2); margin-left:8px; padding:0 16px; border-radius:18px; font:500 14px Roboto,Noto,sans-serif; display:inline-flex; align-items:center; height:36px; text-decoration:none; }
        #${floatBtnID} { background:${btnColor} url("https://iili.io/fObpSDv.png") no-repeat center; background-size:65%; position:fixed; top:70%; right:20px; transform:translateY(-50%); width:60px; height:60px; border-radius:50%; border:none; cursor:grab; display:flex; justify-content:center; align-items:center; z-index:9999; box-shadow:0 6px 12px rgba(0,0,0,.3); transition:opacity .4s ease; opacity:1; }
        #${floatBtnID}.punisher-hidden { opacity:0; pointer-events:none; }
    `);

    const getVideoID = url => {
        const m = url.match(/(?:v=|\/)([0-9A-Za-z_-]{11})|\/shorts\/([0-9A-Za-z_-]{11})/);
        return m ? (m[1] || m[2]) : null;
    };

    const getBypassURL = vid => vid ? punisherURL + vid : null;
    const findButtonContainer = () => {
        return document.querySelector("#top-level-buttons-computed") ||
               document.querySelector("ytd-video-primary-info-renderer #actions") ||
               document.querySelector("ytmusic-player-bar") ||
               document.querySelector("div#menu-container") ||
               document.querySelector("[role='group'][aria-label]");
    };
    let lastVideoId = null;

    async function addPlayerButton() {
        const vid = getVideoID(location.href);
        if (!vid || vid === lastVideoId) return;
        lastVideoId = vid;

        const container = findButtonContainer();
        if (!container) return;

        let btn = document.getElementById(playerBtnID);
        if (!btn) {
            btn = document.createElement("a");
            btn.id = playerBtnID;
            btn.target = "_blank";
            btn.textContent = "Download";
            container.appendChild(btn);
        }
        btn.href = getBypassURL(vid);
    }

    function addFloatButton() {
        if (document.getElementById(floatBtnID)) return;
        const btn = document.createElement("button");
        btn.id = floatBtnID;
        let dragging = false, ox = 0, oy = 0;
        const limit = (x, y) => ({ x: Math.max(0, Math.min(x, innerWidth - btn.offsetWidth)), y: Math.max(0, Math.min(y, innerHeight - btn.offsetHeight)) });
        const open = () => { const vid = getVideoID(location.href); if (vid) window.open(getBypassURL(vid), "_blank"); };
        const start = (x, y) => { dragging = true; const r = btn.getBoundingClientRect(); ox = x - r.left; oy = y - r.top; btn.style.cursor = "grabbing"; };
        const move = (x, y) => { if (!dragging) return; const p = limit(x - ox, y - oy); btn.style.left = p.x + "px"; btn.style.top = p.y + "px"; btn.style.right = btn.style.bottom = "auto"; };
        const end = () => { dragging = false; btn.style.cursor = "grab"; };

        btn.addEventListener("mousedown", e => start(e.clientX, e.clientY));
        document.addEventListener("mousemove", e => move(e.clientX, e.clientY));
        document.addEventListener("mouseup", end);
        btn.addEventListener("touchstart", e => start(e.touches[0].clientX, e.touches[0].clientY), { passive: true });
        document.addEventListener("touchmove", e => move(e.touches[0].clientX, e.touches[0].clientY), { passive: true });
        document.addEventListener("touchend", end);
        btn.addEventListener("click", () => !dragging && open());

        if (window.matchMedia("(pointer: fine)").matches) {
            let hideTimer;
            const showBtn = () => {
                btn.classList.remove("punisher-hidden");
                clearTimeout(hideTimer);
                hideTimer = setTimeout(() => btn.classList.add("punisher-hidden"), 3000);
            };
            document.addEventListener("mousemove", showBtn);
            document.addEventListener("mousedown", showBtn);
            document.addEventListener("keydown", showBtn);
            showBtn();
        }
        document.body.appendChild(btn);
    }

    const update = () => { addPlayerButton(); addFloatButton(); };
    let lastURL = location.href;
    setInterval(() => {
        if (location.href !== lastURL) {
            lastURL = location.href;
            update();
        }
    }, 800);

    const domObserver = new MutationObserver(() => update());
    domObserver.observe(document.body, { childList: true, subtree: true });
    update();
})();