Twitter - Remove Promoted Tweets

Remove promoted tweets from Twitter

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name          Twitter - Remove Promoted Tweets
// @description   Remove promoted tweets from Twitter
// @author        Nick Stakenburg
// @namespace     https://gist.githubusercontent.com/staaky
// @license       MIT; https://opensource.org/licenses/MIT
// @include       http://twitter.com/*
// @include       https://twitter.com/*
// @include       http://*.twitter.com/*
// @include       https://*.twitter.com/*
// @icon          
// @grant         GM_addStyle
// @version       1.1.1
// ==/UserScript==

(() => {
    // remove all tweets that contain an svg path that matches the promoted tweet icon
    // this avoids language based checks
    const pathDs = [
        'M20.75 2H3.25A2.25 2.25 0 0 0 1 4.25v15.5A2.25 2.25 0 0 0 3.25 22h17.5A2.25 2.25 0 0 0 23 19.75V4.25A2.25 2.25 0 0 0 20.75 2zM17.5 13.504a.875.875 0 1 1-1.75-.001V9.967l-7.547 7.546a.875.875 0 0 1-1.238-1.238l7.547-7.547h-3.54a.876.876 0 0 1 .001-1.751h5.65c.483 0 .875.39.875.874v5.65z'
    ];

    // selectors pointing to ad containers
    const newTwitterAdContainerSelector = [
        'article',
        'div[data-focusable="true"]'
    ].map(s => `${s}:not([adschecked])`).join(', ');

    const legacyTwitterAdContainerSelector = [
        'li[data-item-type="tweet"]',
        'li.promoted-trend'
    ].map(s => `${s}:not([adschecked])`).join(', ');

    let checkLegacyTwitter = true;

    function remove(element) {
        // for debugging (blank out instead of remove)
        //element.setAttribute('style', 'opacity: 0');
        element.remove();
    }

    function findElementToRemove(container, iconSelector) {
        let removeElement = null;

        // if the element contains an promoted icon
        if (container.querySelectorAll(iconSelector).length > 0) {
            // return the element iself
            let parent = container.parentNode,
                height = container.getBoundingClientRect().height;

            removeElement = container;

            // or the parent as long as it is the same height as the element (accounting for a border)
            while (parent && (parent.getBoundingClientRect().height <= height + 1)) {
                removeElement = parent;
                parent = parent.parentNode;
            }
        }

        return removeElement;
    }

    function removeAdTweets() {
        // new twitter
        let adContainers = document.querySelectorAll(newTwitterAdContainerSelector);

        for (let container of adContainers) {
            container.setAttribute('adschecked', 'adschecked');

            for (let d of pathDs) {
                let removeElement = findElementToRemove(container, `svg path[d='${d}']`);

                if (removeElement) {
                    remove(removeElement);
                    checkLegacyTwitter = false;

                    break;
                }
            }
        }

        // legacy twitter
        if (checkLegacyTwitter) {
            adContainers = document.querySelectorAll(legacyTwitterAdContainerSelector);

            for (let container of adContainers) {
                container.setAttribute('adschecked', 'adschecked');

                let removeElement = findElementToRemove(container, '.Icon--promoted');

                if (removeElement) {
                    remove(removeElement);
                }
            }
        }
    }

    let mutationObserver = new MutationObserver(removeAdTweets);

    mutationObserver.observe(document.body, { childList: true, subtree: true });

    removeAdTweets();

    if (checkLegacyTwitter) {
        GM_addStyle('.Footer .flex-module.Footer-adsModule { display: none !important; }');
    }
})();