Block Youtube Users

Hide videos of blacklisted users/channels (from recommended, search, related channels...)

Od 20.02.2021.. Pogledajte najnovija verzija.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

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

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

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

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

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

(I already have a user script manager, let me install it!)

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.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         Block Youtube Users
// @author       Schegge
// @namespace    https://github.com/Schegge
// @description  Hide videos of blacklisted users/channels (from recommended, search, related channels...)
// @version      2.4.6
// @match        *://*.youtube.com/*
// @exclude      *://*.youtube.com/embed/*
// @exclude      *://*.youtube.com/live_chat?*
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM.getValue
// @grant        GM.setValue
// @require      https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js
// @require      https://ajax.googleapis.com/ajax/libs/jquery/1.11.3/jquery.min.js
// @icon         
// ==/UserScript==

/** DESCRIPTION
- it is not case-sensitive
- it hides videos of blacklisted users/channels from recommended, search, related channels, comments...
 - also from the playlists/mixes, but it doesn't prevent them from playing if the playlist is in autoplay
- 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, *buzzfeed)
 - when you use it, but still you want continue seeing a channel that has that word in the name, you can put it in the whitelist (example -> balcklist: *buzzfeed; whitelist: BuzzFeed Nifty)
- you can choose the symbol to split the usernames (default is a comma, * and " not allowed, min-max 1 character)
- you can enable/disable to blacklist channels by right clicking on '[x]' before the usernames
 - in any case, the [x] buttons are automatically shown when the "B" menu is open
- from a direct link to youtube, it pauses the video if blacklisted (you can enable/disable it)
- you can suspend temporarily the block (to reactivate it just click on save or refresh the page)
**/

(async function($) {

   /* VARIABLES */

   // get black/whitelist saved
   var sep, add, pv, blacklist, whitelist;
   async function getValues() {
      add = await GM.getValue('enableadd', '');
      pv = await GM.getValue('enablepause', '');
      sep = await GM.getValue('sep', ',');
      blacklist = await GM.getValue('savedblocks', '');
      whitelist = await GM.getValue('savedwhites', '');
      blacklist = blacklist ? blacklist.split(sep).map(function(v) { return v.trim(); }) : [];
      whitelist = whitelist ? whitelist.split(sep).map(function(v) { return v.trim(); }) : [];
   }
   await getValues();

   var byuver = await GM.getValue('byuver', '2.4.6');
   var bOpen = false;
   var suspend = false;

   var uClasses = '' +
      // grid
      '#text.ytd-channel-name, ' +
      // big channel recommend
      '#title-text.ytd-shelf-renderer a.ytd-shelf-renderer[href*="user"], #title-text.ytd-shelf-renderer a.ytd-shelf-renderer[href*="channel"], #title-annotation.ytd-shelf-renderer a, ' +
      // search channels
      '#channel-title.ytd-channel-renderer span.ytd-channel-renderer, ' +
      // related channels
      '.title.ytd-mini-channel-renderer, ' +
      // playlist
      '#byline.ytd-playlist-panel-video-renderer, ' +
      // comment
      '#author-text span.ytd-comment-renderer';
   var tClasses = 'ytd-video-renderer, ytd-grid-video-renderer, ytd-rich-item-renderer, ytd-shelf-renderer, ytd-channel-renderer, ytd-mini-channel-renderer, ytd-playlist-renderer, ytd-compact-video-renderer, ytd-compact-autoplay-renderer, ytd-playlist-panel-video-renderer, ytd-comment-thread-renderer, ytd-movie-renderer, ytd-secondary-search-container-renderer';
   var uVideo = '#upload-info #channel-name #text.ytd-channel-name';

   /* INTERVAL FOR BLACKLISTING */

   search();
   setInterval(search, 1000);

   /* CSS */

   $('head').append('<style> ' +
      '#byu-is-black { display: none!important; } ' +
      '.byu-add { font-size: .8em; margin-right: .5em; cursor: pointer; color: #FF0000; font-family: consolas, monospace; float: left; }' +
      '#byu-video-page-black { position: fixed; z-index: 999999; width: 20%; min-width: 50px; font-size: 1.1em; padding: 1em; bottom: 50px; left: 50px; background: red; color: #fff; border-radius: 2px; }' +
      '#byu { color: var(--yt-spec-icon-active-other); cursor: pointer; font-size: 20px; font-weight: 500; vertical-align: middle; } ' +
      '#byu-options { width: 30%; max-width: 250px; display: flex; flex-flow: row wrap; align-items: baseline; position: fixed; right: 10px; padding: 0 20px 15px; background-color: #fff; box-shadow: 0 1px 2px 0 rgba(0,0,0,.1); border: 1px solid #fafafa; border-top: 0; z-index: 9999999999; } ' +
      '#byu-options div { box-sizing: border-box; padding: 5px; font-size: 1em; } ' +
      '#byu-options .byu-textarea { width: 100%; } ' +
      '#byu-options .byu-textarea div { font-size: 1.2em; width: 100%; text-align: center; font-weight: 500; } ' +
      '#byu-options .byu-textarea textarea { font-size: 1em; line-height: 1em; resize: vertical; width: 100%; padding: 4px; border: 2px solid rgba(0,0,0,.13); box-sizing: border-box; } ' +
      '#byu-options .byu-textarea.bl textarea { min-height: 6em;} ' +
      '#byu-options .byu-textarea.wl textarea { min-height: 4em;} ' +
      '.byu-ver { width: 50%; font-size: .8em; color: rgba(0,0,0,.4); }' +
      '.byu-save { width: 50%; text-align: right; }' +
      '#byu-saveblacklist { font-size: 1.2em; font-weight: bold; cursor: pointer; color: #FF0000; } ' +
      '.byu-sep { width: auto; flex-grow: 1; color: rgba(0,0,0,.5); } ' +
      '.byu-opt { width: auto; flex-grow: 1;  color: rgba(0,0,0,.5); text-align: center; } ' +
      '#byu-suspend { width: 100%; flex-grow: 1;  cursor: pointer; color: rgba(0,0,0,.5); text-align: right; } ' +
      '#byu-sep-symbol { width: 1em; background: #fff; border: 1px solid rgba(0,0,0,.2); padding: 0 2px; color: #000; height: 1.3em; line-height: 1em; vertical-align: bottom; box-sizing: border-box; } ' +
      'input[type="checkbox"] { margin: 0; vertical-align: bottom; } ' +
      '#byu-notice { position: fixed; z-index: 999999; width: 25%; min-width: 200px; font-size: 1.1em; padding: 1.5em; bottom: 50px; right: 50px; color: red; background: #fff; border-radius: .5em; border: 1px solid red; } ' +
      '#byu-notice-dismiss { cursor: pointer; background: red; color: #fff; border-radius: 2px; padding: 0 5px; } ' +
      '</style>');

   /* NEW VERSION NOTIFICATION */

   if (byuver !== '2.4.6') {
      byuver = '2.4.6';
      GM.setValue('byuver', byuver);
      $('body').append('<div id="byu-notice">BLOCK YOUTUBE USERS [' + byuver + ']<br><br>- Now it also hide comments of blacklisted users.<br>- The old youtube layout is no longer supported.<br><br><span id="byu-notice-dismiss">close</span></div>');
      $('#byu-notice-dismiss').on('click', function() { $('#byu-notice').remove(); });
   }

   /* VIDEO FIRST PAGE */

   // when the first page opened is 'watch', pause video if blacklisted
   if (pv && /\/watch/.test(window.location.href)) {
      var video = $('#player video.video-stream.html5-main-video');

      var pausevideo = function(u) {
         if (ifMatch(u.toLowerCase().trim())) {
            video.get(0).pause();
            video.get(0).currentTime = 0;
            var pausing = setInterval(function() {
               if (!suspend && !video.get(0).paused) {
                  video.get(0).pause();
                  video.get(0).currentTime = 0;
               }
            }, 500);
            $('body').append($('<div id="byu-video-page-black">' + u + ' is blacklisted</div>'));
            setTimeout(function() { $('#byu-video-page-black').remove(); clearInterval(pausing); }, 10000);
            $('body').on('click', '.html5-video-player, button.ytp-play-button', function() { clearInterval(pausing); });
         }
      };

      if (typeof window.ytplayer !== 'undefined' && typeof window.ytplayer.config.args.author !== 'undefined') {
         pausevideo(window.ytplayer.config.args.author);
      } else {
         // for greasemonkey
         var waitUvideo = setInterval(function() {
            if ($(uVideo).length) {
               clearInterval(waitUvideo);
               pausevideo($(uVideo).text());
            }
         }, 1000);
      }
   }

   /* BLACKLIST MENU */

   // menu
   $('body').append('<div id="byu-options" style="display: none">' +
      '<div class="byu-ver">v' + byuver + '</div>' +
      '<div class="byu-save"><span id="byu-saveblacklist">save</span></div>' +
      '<div class="byu-textarea bl"><div>Blacklist</div><textarea spellcheck="false" id="byu-blacklist-words">' + blacklist.join(sep + ' ') + '</textarea></div>' +
      '<div class="byu-textarea wl"><div>Whitelist</div><textarea spellcheck="false" id="byu-whitelist-words">' + whitelist.join(sep + ' ') + '</textarea></div>' +
      '<div class="byu-sep">separator <input id="byu-sep-symbol" type="text" value="' + sep + '" maxlength="1" /></div>' +
      '<div class="byu-opt">right-click to add <input type="checkbox" name="clickadd" value="clickadd"' + (add ? ' checked' : '') + '></div>' +
      '<div class="byu-opt">pause blacklisted video <input type="checkbox" name="pausevideo" value="pausevideo"' + (pv ? ' checked' : '') + '></div>' +
      '<div id="byu-suspend">suspend</div>' +
      '</div>');

   // for the button wait till the masthead is added
   var waitButton = setInterval(function() {
      console.log($('#header-bar'), $('#header-bar').length);
      if ($('#buttons').length) {
         clearInterval(waitButton);
         var btn = '<div style="display: inline-block; position: relative; height: 25px; width: 30px"><span id="byu">B</span></div>';
         $('#buttons').before(btn);
         $('head').append('<style>#byu-options {top:' + $('#container.ytd-masthead').height() + 'px; }</style>');
      }
   }, 1000);

   /* BLACKLISTING FUNCTIONS */

   // check if a username is whitelisted
   function ifWhite(u) {
      if (!whitelist) return false;
      return whitelist.some(function(w) {
         return u === w.trim().toLowerCase();
      });
   }

   // check if a username is blacklisted
   function ifBlack(u) {
      return blacklist.some(function(b) {
         if (b.charAt(0) === '*') {
            b = b.replace('*', '');
            return b && u.indexOf(b) !== -1;
         } else {
            b = b.trim().toLowerCase();
            return b && u === b;
         }
      });
   }

   // check if it needs to be blacklisted
   function ifMatch(u) {
      return !suspend && !ifWhite(u) && ifBlack(u);
   }

   // do the thing
   function findMatch(el, newAdd) {
      var username = $(el).text().trim().toLowerCase();
      if (!username) return;
      var same = $(el).data('username') === username;

      if ((!same || newAdd) && ifMatch(username)) {
         $(el).closest(tClasses).attr('id', 'byu-is-black');

      } else if ((add || bOpen) && !$(el).siblings('.byu-add').length) {
         $('<span class="byu-add">[x]</span>').insertBefore($(el));
      }

      if (!same) $(el).data('username', username);
   }

   // global search
   function search(newAdd) {
      $(uClasses).each(function() {
         findMatch($(this), newAdd);
      });
   }

   /* EVENT LISTENERS */

   // open and close options
   $('body').on('click', '#byu', function() {
      $('#byu-options').slideToggle();
      $(this).css('font-weight', $(this).css('font-weight') === '700' ? '500' : '700');
      bOpen = bOpen ? false : true;
      if (!add) {
         if (bOpen) search();
         else $('.byu-add').remove();
      }
   });

   // save blacklist changes and research
   $('#byu-saveblacklist').on('click', async function() {
      if (/[*"]|^$/.test($('#byu-sep-symbol').val())) {
         $(this).text('ERROR! separator not allowed');
      } else {
         // save new values
         await GM.setValue('savedblocks', $('#byu-blacklist-words').val().trim());
         await GM.setValue('savedwhites', $('#byu-whitelist-words').val().trim());
         await GM.setValue('sep', $('#byu-sep-symbol').val());
         // add notification
         $(this).text('saved');
         // clear everything
         $('[id="byu-is-black"]').removeAttr('id');
         suspend = false;
         $('#byu-suspend').css('font-weight', '400');
         // research
         await getValues();
         search(true);
      }
      setTimeout(function() { $('#byu-saveblacklist').text('save'); }, 2000);
   });

   // add usernames to blacklist
   $('body').on('contextmenu', '.byu-add', async function(e) {
      e.preventDefault();
      e.stopPropagation();
      $('#byu-blacklist-words').val($('#byu-blacklist-words').val() + (blacklist.length ? sep + ' ' : '') + $(this).next().data('username'));
      await GM.setValue('savedblocks', $('#byu-blacklist-words').val());
      await getValues();
      search(true);
   });

   // enable/disable click add
   $('input[name=clickadd]').on('change', async function() {
      if ($(this).is(':checked')) {
         add = 'yes';
      } else {
         add = '';
      }
      await GM.setValue('enableadd', add);
   });

   // enable/disable pause video
   $('input[name=pausevideo]').on('change', function() {
      pv = $(this).is(':checked') ? 'yes' : '';
      GM.setValue('enablepause', pv);
   });

   // suspend
   $('#byu-suspend').on('click', function() {
      suspend = true;
      $('[id="byu-is-black"]').removeAttr('id');
      $(this).css('font-weight', '700');
   });

})(jQuery);