// ==UserScript== // @name Block Youtube Users // @author Schegge // @namespace https://github.com/Schegge // @description Prevent from seeing videos by certain users (from recommended, search, related channels...) // @version 2.1.8 // @match *://www.youtube.com/* // @exclude *://www.youtube.com/embed/* // @require https://ajax.googleapis.com/ajax/libs/jquery/1.11.3/jquery.min.js // @grant GM_getValue // @grant GM_setValue // @icon  // @downloadURL none // ==/UserScript== /**************************************** → the program is case-insensitive → split the usernames with a comma → put a * in front of a word for wildcard (only in the blacklist!), it will find the word no matter its position in the username (example: *vevo) → it also hides videos from the playlists/mixes, but it doesn't prevent them from playing if the playlist is in autoplay (download another script that disables autoplay!) → VERSION 2: # blacklist editor changed # whitelist added # no need to refresh the page after changes in the blacklist/whitelist # added ☆ button to force a new search (in case of problems?) # added button for suspend temporarily the block (to reactivate it just click on the star/save or refresh the page) please report any bugs ****************************************/ (function($) { /*/ DEBUGGING ///////////////////////////////////////////////////////////////////////////// function debugging(varName, variable, parents) { var message = "{BLACKLIST} [" + varName + "]"; if ( variable !== undefined ) message += " (" + typeof variable + ") " + variable; console.log( message ); if ( parents !== undefined ) console.log( $(parents).parents() ); } /*///////////////////////////////////////////////////////////////////////////////////////// // get black/whitelist saved var sBL, sWL, ytblacklist, ytwhitelist; function getValues() { sBL = GM_getValue("savedblocks", "the program is case-insensitive, split the usernames with a comma, put a * in front of a word for wildcard, it will find the word no matter its position in the username, example, *vevo, delete all of this"); // debugging("getValue] [Blacklist", sBL); sWL = GM_getValue("savedwhites", "put here whitelisted usernames, if for example you blacklist *vevo, but you want to see IndilaVEVO, write here:, indilavevo, delete all of this"); // debugging("getValue] [Whitelist", sWL); ytblacklist = sBL.split(","); ytwhitelist = sWL.split(","); } getValues(); // where the usernames are var uClasses = [".g-hovercard", ".branded-page-related-channels-list", ".branded-page-module-title-text", ".video-uploader-byline"]; // add blacklist button to masthead $buttonB = $("", { id: "yt-blacklist", html: "B", css: { "cursor": "pointer", "margin-right": "2px", "font-size": "22px", "vertical-align": "middle" } }); $buttonR = $("", { id: "yt-blacklist-research", html: "☆", css: { "cursor": "pointer", "font-size": "12px", "vertical-align": "top", "margin-right": "5px" } }); $("").insertAfter("#upload-btn") .append($buttonB) .append($buttonR); // elements for user input var marginright = ( $(window).width() - $buttonB.offset().left - $buttonB.outerWidth() - $buttonR.outerWidth() - 20 ); var margintop = $("#yt-masthead-container").height() + parseInt($("#yt-masthead-container").css("padding-top")) + parseInt($("#yt-masthead-container").css("padding-bottom")); $divInput = $("
", { id: "yt-blacklist-options", css: { "display": "none", "position": "fixed", "right": marginright + "px", "top": margintop + "px", "padding": "0 20px 15px 20px", "text-align": "center", "background-color": "#fff", "box-shadow": "0 1px 1px 0 rgba(0,0,0,.1)", "border": "1px solid #e8e8e8", "border-top": "0", "z-index": "99999999999" } }); $textareaBL = $('
' + '
Blacklist
' + '
'); $textareaWL = $('
' + '
Whitelist
' + '
'); $saveDiv = $('
' + '' + ' save
'); $suspend = $("
suspend block
"); $saved = $(' saved and searched again '); $divInput .append($saveDiv) .append($textareaWL) .append($textareaBL) .append($suspend); $("body").append($divInput); // open and close textareas $("#yt-blacklist").click(function() { $("#yt-blacklist-options").slideToggle(); }); // check if a username is whitelisted function ifWhite(u) { var whitelisted = false; for(var z = 0; z < ytwhitelist.length; z++) { var w = ytwhitelist[z].trim().toLowerCase(); if (w.length && u === w) { whitelisted = true; } } // debugging(u + " white?", whitelisted); return whitelisted; } // check if a username is blacklisted function ifMatch(u) { var match = false; if ( !ifWhite(u) ) { // if the username isn't whitelisted for (var j = 0; j < ytblacklist.length; j++) { var b = ytblacklist[j].trim().toLowerCase(); if ( b.charAt(0) == "*" ) { // wildcards var part = b.split("*"), item = part[1]; if ( item.length && u.indexOf(item) !== -1 ) { match = true; } } else { // exact match if ( b.length && u == b ) { match = true; } } } } // debugging(u + " black?", match); return match; } // delete blacklisted function suspend(s) { $(s).each(function() { if ( $(this).siblings(".span-is-black").length ) { $(this).siblings(".span-is-black").remove(); } else if ( $(this).parents(".li-is-black").length ) { $(this).parents(".li-is-black").removeClass("li-is-black").show(); } else if ( $(this).parents(".tr-is-black").length ) { $(this).parents(".tr-is-black").removeClass("tr-is-black").show(); } }); } // do the thing function findMatch(s) { $(s).each(function() { var username = $(this).text().trim().toLowerCase(); //debugging(username + " | parents:", undefined, this); if ( !username ) return 'continue'; if ( ifMatch(username) ) { // if the username is blacklisted if ( $(this).parents("#watch-header").length ) { // WATCH VIDEO if ( !$(this).siblings(".span-is-black").length ) { // check if it wasn't already blacklisted $(".yt-user-info").append("BLACKLISTED!"); } } else if ( $(this).parents("tr.pl-video").length ) { // PLAYLIST (not the 'dark' ones) if ( !$(this).parents(".tr-is-black").length ) { $(this).closest("tr").addClass("tr-is-black").hide(); } } else { // SEARCH, RECOMMENDED, etc... if ( !$(this).parents(".li-is-black").length ) { $(this).closest("li").addClass("li-is-black").hide(); } } } else { // if a previous black/whitelist word is deleted/added suspend(this); } }); } // the final search function function search() { var url = window.location.href; // playlist?list=WL = Watch Later - feed/... = History, Subscriptions if ( !/.*youtube\.com\/(playlist\?list=WL|feed\/\w+$)/.test(url) ) { for (var i = 0; i < uClasses.length; i++) { findMatch(uClasses[i]); } } } // save blacklist changes and research $("#saveblacklist").click(function() { GM_setValue("savedblocks", $('#blacklist-words').val()); GM_setValue("savedwhites", $('#whitelist-words').val()); getValues(); search(); $(this).before($saved); setTimeout(function() { $saved.remove(); }, 2000); }); // research when $buttonR is clicked $("#yt-blacklist-research").click(function() { search(); }); // suspend the block when $suspend is clicked $("#yt-blacklist-suspend").click(function() { for (var i = 0; i < uClasses.length; i++) { suspend(uClasses[i]); } }); // search when youtube is first opened search(); // research after every change in #content var target = document.querySelector('#content'); var config = { attributes: true, childList: true, characterData: true, subtree: true }; var observer = new MutationObserver(function(mutations) { //debugging("MUTATIONS! NEW SEARCH"); search(); }); try { observer.observe(target, config); } catch (e) {} })(jQuery);