ServusTV Adblocker

Blocks Banner, Preroll & Midroll Ads on ServusTV.com

K instalaci tototo skriptu si budete muset nainstalovat rozšíření jako Tampermonkey, Greasemonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Userscripts.

You will need to install an extension such as Tampermonkey to install this script.

K instalaci tohoto skriptu si budete muset nainstalovat manažer uživatelských skriptů.

(Už mám manažer uživatelských skriptů, nechte mě ho nainstalovat!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(Už mám manažer uživatelských stylů, nechte mě ho nainstalovat!)

// ==UserScript==
// @name            ServusTV Adblocker
// @name:de         ServusTV Werbeblocker
// @version         1.0.4
// @description     Blocks Banner, Preroll & Midroll Ads on ServusTV.com
// @description:de  Blockiert Banner, Preroll & Midroll Werbungen auf ServusTV.com
// @icon            https://external-content.duckduckgo.com/ip3/www.servustv.com.ico
// @author          TalkLounge (https://github.com/TalkLounge)
// @namespace       https://github.com/TalkLounge/servustv-adblocker
// @license         MIT
// @match           https://www.servustv.com/*
// @match           https://www.m3u8player.online/embed/*
// @grant           unsafeWindow
// ==/UserScript==

(function () {
    'use strict';
    let lastURL;

    async function init() {
        [...document.querySelectorAll(".ad-wrap, [id^=adslot]")].forEach(item => item.parentNode?.remove());

        if (lastURL == window.location.pathname) {
            return;
        }

        lastURL = window.location.pathname;

        const url = window.location.pathname.split("/").filter(item => item).reverse();
        if (url[1] != "v") {
            return;
        }

        while (true) {
            if (document.querySelector(".theatre-mode>div>picture") && document.querySelector(".theatre-mode>div>div + div div")) {
                break;
            }

            await new Promise(r => setTimeout(r, 100));
        }

        document.querySelector(".theatre-mode>div>picture").style.display = "none";
        document.querySelector(".theatre-mode>div>div").style.display = "none";

        document.querySelector(".theatre-mode>div>div + div div").style.display = "none";
        document.querySelector(".theatre-mode>div>div + div rbup-video-stv")?.remove();
        document.querySelector(".theatre-mode>div>div + div iframe")?.remove();

        let data = await fetch(`https://api-player.redbull.com/stv/servus-tv-playnet?videoId=${url[0].toUpperCase()}`, {
            "method": "GET",
            "referrer": "https://www.servustv.com/"
        });
        data = await data.json();

        const iframe = document.createElement("iframe");
        iframe.src = `https://www.m3u8player.online/embed/dash?url=${encodeURIComponent(data.videoUrl)}`;
        iframe.width = "100%";
        iframe.height = "100%";
        iframe.style.border = "none";
        iframe.style.aspectRatio = "16/9";
        iframe.allowFullscreen = true;

        document.querySelector(".theatre-mode>div>div + div").append(iframe);
    }

    async function deleteCookieBanner() {
        for (let i = 0; i < 4 * 10; i++) {
            if (document.querySelector(".fixed")) {
                document.querySelector(".fixed").remove();
                break;
            }

            await new Promise(r => setTimeout(r, 250));
        }
    }

    if (window.location.href.startsWith("https://www.servustv.com/") && window.top == window.self) { // Main Page
        window.setInterval(init, 250);
    } else if (window.location.href.startsWith("https://www.m3u8player.online/embed/m3u8") && window.top != window.self) { // Iframe
        deleteCookieBanner();
    }

    /*
    Gemini 2.5 Pro is unreal. Just pasted this prompt:
    Can you find in this source code of this page any adblock detection script? If so, create a userscript which blocks it. The page is a single page application so it doesn't work to just block redirection or something.

    And two complete minified source files from the webpage where I thought it would contain the Anti-Adblock mechanism

    And the code I got worked flawless on the first attempt
    */
    Object.defineProperties(unsafeWindow, {
        google_tag_manager: {
            value: {},
            writable: false,
            configurable: false
        },
        OneTrust: {
            value: {
                IsAlertBoxClosed: () => true,
                RejectAll: () => { },
                AllowAll: () => { },
            },
            writable: true,
            configurable: true
        }
    });

    const originalFetch = unsafeWindow.fetch;
    unsafeWindow.fetch = function (url, options) {
        const urlString = url instanceof Request ? url.url : String(url);
        if (['securepubads', 'pagead2', 'doubleclick.net', 'google-analytics', 'fundingchoicesmessages', 'googleadservices', 'googlesyndication'].some(domain => urlString.includes(domain))) {
            return Promise.resolve(new Response(null, { status: 200, statusText: "OK" }));
        }
        return originalFetch.apply(this, arguments);
    };

    try {
        const originalOffsetParent = Object.getOwnPropertyDescriptor(unsafeWindow.Element.prototype, 'offsetParent');
        Object.defineProperty(unsafeWindow.Element.prototype, 'offsetParent', {
            get() {
                if (['ad-unit', 'advertisement', 'adsbox', 'stv-analytics'].some(cls => this.classList.contains(cls))) {
                    return document.body;
                }
                if (originalOffsetParent && originalOffsetParent.get) {
                    return originalOffsetParent.get.call(this);
                }
                return this.parentNode;
            },
            configurable: true
        });
    } catch (e) { }
})();