4chan Media-B-Gone

Filter media on 4chan using duplicate detection

Tendrás que instalar una extensión para tu navegador como Tampermonkey, Greasemonkey o Violentmonkey si quieres utilizar este script.

Necesitarás instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Userscripts para instalar este script.

Necesitará instalar una extensión como Tampermonkey para instalar este script.

Necesitarás instalar una extensión para administrar scripts de usuario si quieres instalar este script.

(Ya tengo un administrador de scripts de usuario, déjame instalarlo)

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

(Ya tengo un administrador de estilos de usuario, déjame instalarlo)

// ==UserScript==
// @name         4chan Media-B-Gone
// @namespace    https://boards.4chan.org/
// @version      1.2.2
// @license      GPLv3
// @description  Filter media on 4chan using duplicate detection
// @author       ceodoe
// @match        https://boards.4chan.org/*/thread/*
// @match        https://boards.4chan.org/*/res/*
// @match        https://boards.4chan.org/*/catalog
// @match        https://boards.4chan.org/*/
// @icon         
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_addStyle
// ==/UserScript==

// Load settings
let filteredImageHashes = GM_getValue("filteredImageHashes", []).filter(e => e.trim() !== "");
let settings = GM_getValue("settings", {
    use4chanX: true,
    hideCompletely: true,
    addTo4chanXFilter: true,
    addTo4chanXFilterNoStub: true,
    hammingDistance: 5,
    nukeBtnText: "☢️",
    showHashInstead: false
});

function main() {
    GM_addStyle(`
        .hidden {
            display: none !important;
        }

        span#fcmbg-settings-btn {
            cursor: pointer;
            user-select: none;
        }

        dialog#fcmbg-settings-dialog::backdrop {
            backdrop-filter: blur(5px);
        }

        dialog#fcmbg-settings-dialog h3 {
            margin: 0;
            margin-bottom: 0.5em;
        }

        dialog#fcmbg-settings-dialog {
            box-sizing: border-box;
            color: inherit;
            background-color: inherit;
            font-family: inherit;
            border-radius: 10px;
            border: 1px solid #888;
            min-width: 40em;
            max-width: 50%;
        }

        span#fcmbg-closebtn {
            cursor: pointer;
            float: right;
        }

        textarea#fcmbg-hashes-textbox {
            width: 98%;
            width: -moz-available;
            width: -webkit-fill-available;
            height: 100px;
            margin-bottom: 0.5em;
            color: inherit;
            background-color: inherit;
        }

        input.fcmbg-indented-checkbox {
            margin-left: 2em;
        }

        input.fcmbg-indented-checkbox-2 {
            margin-left: 4em;
        }

        input[type=checkbox]:disabled + label {
            color: #888;
        }

        .fcmbg-positive {
            color: #0f0;
        }

        .fcmbg-negative {
            color: #f00;
        }

        .fcmbg-positive, .fcmbg-negative {
            font-weight: bold;
        }

        div#fcmbg-save-button-wrapper {
            width: 98%;
            width: -moz-available;
            width: -webkit-fill-available;
            margin: auto;
            margin-top: 2em;
            text-align: center;
        }

        .fcmbg-footnote {
            margin-bottom: 0.2em;
        }

        .fcmbg-nuke-button {
            margin-left: 0.2em;
            cursor: pointer;
            user-select: none;
        }
    `);

    window.addEventListener("load", function () {
        let settingsDialog = this.document.createElement("dialog");
        settingsDialog.id = "fcmbg-settings-dialog";
        settingsDialog.innerHTML = `
            <span id="fcmbg-closebtn" title="Close" onclick="this.closest('dialog').close()">❌</span>
            <h3>☢️ 4chan Media-B-Gone</h3>

            Filtered thumbnail hashes (comma-separated) <sup>[1]</sup> :<br />
            <textarea id="fcmbg-hashes-textbox" placeholder="List of comma-separated hashes">${filteredImageHashes.join(",")}</textarea><br />

            <input type="checkbox" id="fcmbg-checkbox-usefourchanx" ${settings.use4chanX ? "checked" : ""}><label for="fcmbg-checkbox-usefourchanx">Integrate with 4chan X (${is4chanXActive() ? "<span class='fcmbg-positive'>Detected</span>" : "<span class='fcmbg-negative'>Not detected</span>"})</label><br />
                <input type="checkbox" id="fcmbg-checkbox-hidecompletely" class="fcmbg-indented-checkbox" ${settings.hideCompletely ? "checked" : ""}><label for="fcmbg-checkbox-hidecompletely">Hide matched media completely instead of leaving a stub <sup>[2]</sup></label><br />
                <input type="checkbox" id="fcmbg-checkbox-add-to-filter" class="fcmbg-indented-checkbox" ${settings.addTo4chanXFilter ? "checked" : ""}><label for="fcmbg-checkbox-add-to-filter">Add matched media to 4chan X's Image MD5 filter</label><br />
                <input type="checkbox" id="fcmbg-checkbox-add-to-filter-nostub" class="fcmbg-indented-checkbox-2" ${settings.addTo4chanXFilterNoStub ? "checked" : ""}><label for="fcmbg-checkbox-add-to-filter-nostub">Add <code>stub:no;</code> to generated 4chan X filters to override stub setting <sup>[3]</sup></label><br /><br />

            <input type="checkbox" id="fcmbg-checkbox-hash-instead-of-btn" ${settings.showHashInstead ? "checked" : ""}><label for="fcmbg-checkbox-hash-instead-of-btn">Show copiable media hash on posts instead of nuke button</label><br />
            Text to show on the hide button: <input type="text" id="fcmbg-checkbox-nuke-button-text" value="${settings.nukeBtnText || "☢️"}" /><br />

            <br />

            Hamming distance <sup>[4]</sup> : <input type="number" value="${settings.hammingDistance}" max="15" min="0" id="fcmbg-hamming-distance-numeric" /><br /></<br /><br />

            <small>
                <div class="fcmbg-footnote">
                    <sup>[1]</sup> Note that these are <i>average hashes</i>, not MD5 hashes, they are incompatible with 4chan X's hashes
                </div>
                <div class="fcmbg-footnote">
                    <sup>[2]</sup> If stubs are turned off in 4chan X's settings, this setting will have no effect.
                </div>
                <div class="fcmbg-footnote">
                    <sup>[3]</sup> This requires a page reload to take effect since 4chan X doesn't reparse posts unless the page is reloaded.
                </div>
                <div class="fcmbg-footnote">
                    <sup>[4]</sup> Valid values are 0-15. A value of 0 means no tolerance for any differences in the image, a value of 15 means a high tolerance for differences. A value of 5 is ideal.
                </div>
            </small>

            <div id="fcmbg-save-button-wrapper">
                <input type="button" value="Save and reload" id="fcmbg-save-button" />
            </div>
        `;

        document.body.insertAdjacentElement("afterbegin", settingsDialog);

        // Disable all 4chan X related options if it's not detected
        if(!is4chanXActive()) {
            document.querySelector("#fcmbg-checkbox-usefourchanx").disabled = true;
            document.querySelector("#fcmbg-checkbox-hidecompletely").disabled = true;
            document.querySelector("#fcmbg-checkbox-add-to-filter").disabled = true;
            document.querySelector("#fcmbg-checkbox-add-to-filter-nostub").disabled = true;
        }

        document.querySelector("#fcmbg-checkbox-usefourchanx").addEventListener("change", function() {
            if(this.checked) {
                document.querySelector("#fcmbg-checkbox-hidecompletely").disabled = false;
                document.querySelector("#fcmbg-checkbox-add-to-filter").disabled = false;
                document.querySelector("#fcmbg-checkbox-add-to-filter-nostub").disabled = false;
            } else {
                document.querySelector("#fcmbg-checkbox-hidecompletely").disabled = true;
                document.querySelector("#fcmbg-checkbox-add-to-filter").disabled = true;
                document.querySelector("#fcmbg-checkbox-add-to-filter-nostub").disabled = true;
            }
        });

        document.querySelector("#fcmbg-save-button").addEventListener("click", function() {
            settings.use4chanX = document.querySelector("#fcmbg-checkbox-usefourchanx").checked;
            settings.showHashInstead = document.querySelector("#fcmbg-checkbox-hash-instead-of-btn").checked;
            settings.hideCompletely = document.querySelector("#fcmbg-checkbox-hidecompletely").checked;
            settings.addTo4chanXFilter = document.querySelector("#fcmbg-checkbox-add-to-filter").checked;
            settings.addTo4chanXFilterNoStub = document.querySelector("#fcmbg-checkbox-add-to-filter-nostub").checked;
            settings.nukeBtnText = document.querySelector("#fcmbg-checkbox-nuke-button-text").value;
            settings.hammingDistance = document.querySelector("#fcmbg-hamming-distance-numeric").value;
            GM_setValue("settings", settings);

            // Do not reload hashes since user may have edited the list directly
            filteredImageHashes = document.querySelector("#fcmbg-hashes-textbox").value.split(",").map(e => e.trim()).filter(e => e !== "");

            GM_setValue("filteredImageHashes", [...new Set(filteredImageHashes.filter(e => e.trim() !== ""))]);

            location.reload();
        });

        settingsDialog.addEventListener("click", function(event) {
            let rect = this.getBoundingClientRect();
            let isInDialog = (
                rect.top <= event.clientY &&
                event.clientY <= rect.top + rect.height &&
                rect.left <= event.clientX &&
                event.clientX <= rect.left + rect.width
            );

            if(!isInDialog) {
                this.close();
            }
        });


        if(is4chanXActive() && settings.use4chanX) {
            let fcMBGSettingsBtn = document.createElement("span");
            fcMBGSettingsBtn.innerText = "☢️";
            fcMBGSettingsBtn.id = "fcmbg-settings-btn";

            let fcXSettingsBtn = document.querySelector("span#shortcut-settings");
            fcMBGSettingsBtn.classList.add("shortcut");
            fcMBGSettingsBtn.classList.add("brackets-wrap");
            fcXSettingsBtn.insertAdjacentElement("afterend", fcMBGSettingsBtn);
        } else {
            // Thread
            let anchorElem = document.querySelector("div.navLinks.desktop");

            // Index
            if(!anchorElem) {
                anchorElem = document.querySelector("div#ctrl-top");
            }

            // Catalog
            if(!anchorElem) {
                anchorElem = document.querySelector("span.navLinks");
            }

            anchorElem.insertAdjacentHTML("beforeend", `
                [ <a id="fcmbg-settings-btn" href="#">☢️ Media-B-Gone</a> ]
            `);
        }

        document.querySelector("#fcmbg-settings-btn").onclick = function() {
            document.querySelector("#fcmbg-settings-dialog").showModal();
        };

        // Skip initial load if we load into 4chan X catalog, and let the MutationObserver deal with the posts
        if(!document.querySelector("div.board.catalog-small, div.board.catalog-large")) {
            let images = document.querySelectorAll("a.fileThumb > img, a.catalog-link > img.catalog-thumb, div.thread img.thumb");
            let md5Hashes4chanX = [];

            // Mark initial batch parsed before starting observer so we don't process twice
            images.forEach(img => {
                (img.closest("div.postContainer") || img.closest("div.thread")).setAttribute("data-fcmbg-parsed", "1");
            });

            let initialLoop = new Promise((resolve, reject) => {
                let promises = Array.from(images).map(img => {
                    return getImageHash(img.src).then(hash => {
                        post = (img.closest("div.postContainer") || img.closest("div.thread"));

                        if(settings.showHashInstead) {
                            let hashElem = this.document.createElement("span");
                            hashElem.innerText = " " + hash + " ";
                            hashElem.title = "Media hash for Media-B-Gone";

                            let anchorElem = (post.querySelector("div.catalog-stats") || post.querySelector("span.postNum.desktop") || post.querySelector("div.meta"));
                            anchorElem.insertAdjacentElement("afterend", hashElem);
                        }

                        img.setAttribute("data-fcmbg-ahash", hash);
                        for(let i = 0; i < filteredImageHashes.length; i++) {
                            if(hammingDistance(hash, filteredImageHashes[i]) <= settings.hammingDistance) {
                                // Adding an all-zero hash will nuke a ton of unintended media, it represents an image whose pixels are close to being all the same
                                if(hash !== "0000000000000000") {
                                    nukePost(hash, post);
                                    if(settings.use4chanX) {
                                        if(is4chanXActive() && settings.addTo4chanXFilter) {
                                            md5Hashes4chanX.push(img.getAttribute("data-md5"));
                                        }
                                    }
                                    break; // no need to try to hide already hidden post
                                }
                            }
                        }
                    });
                });

                Promise.all(promises).then(resolve).catch(reject);
            });

            initialLoop.then(() => {
                if(is4chanXActive() && settings.addTo4chanXFilter) {
                    if(md5Hashes4chanX.length > 0) {
                        update4chanXFilter(md5Hashes4chanX)
                    }
                }
            });

            for(let i = 0; i < images.length; i++) {
                if(images[i].classList.contains("catalog-thumb")) {
                    addNukeButton(images[i].closest("div.catalog-thread"));
                } else if(images[i].closest("div.postContainer")){
                    addNukeButton(images[i].closest("div.postContainer"));
                } else {
                    addNukeButton(images[i].closest("div.thread"));
                }
            }
        }
    });

    new MutationObserver(function(event) { parseNewPosts(); }).observe(document.querySelector("div.thread") || document.querySelector("div.board"), {subtree: true, childList: true});
}

function parseNewPosts() {
    let images = document.querySelectorAll("a.fileThumb > img, a.catalog-link > img.catalog-thumb, div.thread img.thumb");

    for(let i = 0; i < images.length; i++) {
        let post = (images[i].closest("div.catalog-thread") || images[i].closest("div.thread") || images[i].closest("div.postContainer"));
        if(post.getAttribute("data-fcmbg-parsed")) {
            continue;
        } else {
            post.setAttribute("data-fcmbg-parsed", "1");
            addNukeButton(post);
            processImage(images[i]);
        }
    }
}

async function processImage(img) {
    let hash = await getImageHash(img.src);
    img.setAttribute("data-fcmbg-ahash", hash);

    for(let i = 0; i < filteredImageHashes.length; i++) {
        if(hammingDistance(hash, filteredImageHashes[i]) <= settings.hammingDistance) {
            nukePost(hash, img.closest("div.postContainer"));

            if(is4chanXActive() && settings.addTo4chanXFilter) {
                update4chanXFilter(img.getAttribute("data-md5"));
            }

            break;
        }
    }
}

function addNukeButton(post) {
    if(settings.showHashInstead) {
        return;
    }

    // Try to ascend to an ancestor to avoid duping nuke buttons
    let ancestor = post.closest("div.postContainer") || post;
    if(!ancestor.querySelector(".fcmbg-nuke-button")) {
        post = ancestor;
    } else {
        return;
    }

    let nukeBtn = document.createElement("span");

    nukeBtn.innerText = settings.nukeBtnText || "☢️";
    nukeBtn.title = "Nuke media with 4chan Media-B-Gone";
    nukeBtn.classList.add("fcmbg-nuke-button");
    nukeBtn.addEventListener("click", function() {
        let img = post.querySelector("a.fileThumb > img, a.catalog-link > img.catalog-thumb, div.thread img.thumb");
        let aHash = img.getAttribute("data-fcmbg-ahash");
        let md5hash = img.getAttribute("data-md5");

        if(hammingDistance(aHash, "0000000000000000") <= settings.hammingDistance) {
            alert("You are trying to hide a piece of media which is very close to has to all zeroes. This has unintended consequences and will lead to a lot of false positives when it comes to videos that start with a black or white frame.");
        } else {
            nukePost(aHash, post);

            if(is4chanXActive() && settings.addTo4chanXFilter) {
                update4chanXFilter(md5hash);
            }
        }
    });

    let anchorElem = (post.querySelector("div.catalog-stats") || post.querySelector("span.postNum.desktop") || post.querySelector("div.meta"));
    anchorElem.insertAdjacentElement("afterend", nukeBtn);
}

function nukePost(hash, post) {
    // Refresh hashes in case they were updated in another tab
    filteredImageHashes = GM_getValue("filteredImageHashes", []);
    filteredImageHashes.push(hash);
    document.querySelector("#fcmbg-hashes-textbox").value = filteredImageHashes.join(",");
    GM_setValue("filteredImageHashes", [...new Set(filteredImageHashes.filter(e => e.trim() !== ""))]);
    document.querySelector("textarea#fcmbg-hashes-textbox").value = filteredImageHashes.join(",");

    if(is4chanXActive() && settings.use4chanX) {
        if(post && !post.querySelector("div.stub")) {
            let hideBtn = post.querySelector("a.hide-reply-button");
            if(hideBtn) {
                hideBtn.click();
            }
        }

        if(post && ((is4chanXActive() && settings.hideCompletely) || is4chanXActive() == false)) {
            post.classList.add("hidden");
        }
    } else {
        if(post) {
            post.classList.add("hidden");
        }
    }
}

function update4chanXFilter(newHashes) {
    if(newHashes.constructor == Array) {
        newHashes = newHashes.map(function(e) {
            return `/${e}/${settings.addTo4chanXFilterNoStub ? ";stub:no;" : ""}`;
        });
    } else if(typeof newHashes == "string") {
        newHashes = [`/${newHashes}/${settings.addTo4chanXFilterNoStub ? ";stub:no;" : ""}`];
    }


    document.querySelector("a.settings-link").click();
    document.querySelector("div#overlay").classList.add("hidden");
    document.querySelector("a.tab-filter").click();
    document.querySelector("select[name=filter]").value = "MD5";
    document.querySelector("select[name=filter]").dispatchEvent(new Event("change"));

    let waitFor4chanX = window.setInterval(function() {
        let md5field = document.querySelector("textarea.field[name=MD5]");

        if(md5field) {
            window.clearInterval(waitFor4chanX);
            let md5s = md5field.value.split("\n").map(function(e) { return e.trim(); });
            md5s = md5s.concat(newHashes);

            // Remove dupes
            md5s = [...new Set(md5s)];

            md5field.value = md5s.join("\n");
            md5field.dispatchEvent(new Event("change"));

            document.querySelector("div#overlay").classList.remove("hidden");
            document.querySelector("a.close.fa.fa-times").click();
        }
    }, 100);
}

async function getImageHash(url) {
    return new Promise((resolve, reject) => {
        let img = new Image();
        img.crossOrigin = "Anonymous";
        img.onload = () => {
            let canvas = document.createElement("canvas");
            let ctx = canvas.getContext("2d");
            canvas.width = 8;
            canvas.height = 8;
            ctx.drawImage(img, 0, 0, 8, 8);

            let imageData = ctx.getImageData(0, 0, 8, 8).data;
            let grayValues = [];

            // Convert to grayscale
            for(let i = 0; i < imageData.length; i += 4) {
                let r = imageData[i];
                let g = imageData[i + 1];
                let b = imageData[i + 2];

                // Grayscale average
                let gray = Math.round((r + g + b) / 3);
                grayValues.push(gray);
            }

            // Compute average brightness
            let avg = grayValues.reduce((sum, val) => sum + val, 0) / grayValues.length;

            // Build hash: 1 if pixel > avg, else 0
            let bits = grayValues.map(v => (v > avg ? 1 : 0)).join('');

            // Convert bits to hex string
            let hash = parseInt(bits, 2).toString(16).padStart(16, '0');

            resolve(hash);
        };
        img.onerror = () => reject(new Error("Failed to load image"));
        img.src = url;
    });
}

function hexToBinary(hex) {
    return hex.split("").map(h =>
        parseInt(h, 16).toString(2).padStart(4, "0")
    ).join("");
}

function hammingDistance(hash1, hash2) {
    let bin1 = hexToBinary(hash1);
    let bin2 = hexToBinary(hash2);

    let dist = 0;
    for(let i = 0; i < bin1.length; i++) {
        if(bin1[i] !== bin2[i]) {
            dist++;
        }
    }
    return dist;
}

function is4chanXActive() {
    return document.documentElement.classList.contains("fourchan-x");
}

main();