Block Youtube Users

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

Från och med 2018-04-07. Se den senaste versionen.

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.1
// @match        *://www.youtube.com/*
// @exclude      *://www.youtube.com/embed/*
// @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
  - for both the new and the old youtube layout
  - it is case-insensitive
  - it hides videos of blacklisted users/channels from recommended, search, related channels...
  - also from the playlists/mixes, but it doesn't prevent them from playing if the playlist is in autoplay
  - from a direct link to youtube, it pauses the video if blacklisted (you can enable/disable it)
  - 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)
  - you can choose the symbol to split the usernames (default is a comma, '*' not allowed, max 1 character)
  - you can enable/disable to blacklist channels by clicking (old yt layout)/right clicking (new yt layout) on '[x]' before the usernames
  - you can suspend temporarily the block (to reactivate it just click on save or refresh the page)
***/

(async function($) {

	// saved data from old localStorage if it exits
	var _sBL, _sWL, _sep, _add;
	if (localStorage.getItem('byuver')) {
		_sBL = localStorage.getItem('savedblocks'); localStorage.removeItem('savedblocks');
		_sWL = localStorage.getItem('savedwhites'); localStorage.removeItem('savedwhites');
		_sep = localStorage.getItem('sep');         localStorage.removeItem('sep');
		_add = localStorage.getItem('enableadd');   localStorage.removeItem('enableadd');
		localStorage.removeItem('byuver');
	// default values
	} else {
		_sBL = '';
		_sWL = '';
		_sep = ',';
		_add = '';
	}

	// get black/whitelist saved
	var sBL, sWL, sep, add, pv, ytblacklist, ytwhitelist;
	async function getValues() {
		sBL = await GM.getValue('savedblocks', _sBL);
		sWL = await GM.getValue('savedwhites', _sWL);
		sep = await GM.getValue('sep', _sep);
		add = await GM.getValue('enableadd', _add);
		pv = await GM.getValue('enablepause', '');
		ytblacklist = sBL.split(sep);
		ytwhitelist = sWL.split(sep);
	}
	await getValues();

	// vars
	var suspend = false;
	var uClasses, tClasses, uVideo, margintop;

	// check what layout
	var ver = $('#upload-btn').length ? 'old' : 'new';

	if (ver === 'new') {
		// where the usernames are
		//          //grid
		uClasses = ['#byline.ytd-grid-video-renderer a',
						//big channel recommend
						'a.ytd-shelf-renderer[href*=user] #title.ytd-shelf-renderer, a.ytd-shelf-renderer[href*=channel] #title.ytd-shelf-renderer, #title-annotation.ytd-shelf-renderer a',
						//search
						'#byline.ytd-video-meta-block',
						//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'];
		tClasses = 'ytd-video-renderer, ytd-grid-video-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';
		uVideo = '#owner-name a';

		// research
		window.addEventListener('yt-action', search, false);
		window.addEventListener('yt-page-data-updated', search, false);
		window.addEventListener('yt-load-next-continuation', search, false);
		window.addEventListener('yt-load-reload-continuation', search, false);
		window.addEventListener('shown-items-changed', search, false);

	} else { // old
		uClasses = ['.yt-lockup-byline > a',
						'.branded-page-module-title-text',
						'span.shelf-annotation.shelf-title-annotation a',
						'span.stat.attribution > span:not(.byu-add)',
						'.branded-page-related-channels-list .yt-uix-tile-link',
						'.video-uploader-byline'];
		tClasses = 'tr, li';
		uVideo = '.yt-user-info a';

		// research
		var target = document.querySelector('#content');
		var config = { attributes: false, childList: true, characterData: false, subtree: true };
		var observer = new MutationObserver(function(mutations) { search(); });

		try { observer.observe(target, config); } catch (e) {}
	}

	// 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; vertical-align: top; }' +

		'#byu-notice { position: fixed; z-index: 999999; width: 35%; min-width: 200px; font-size: 1.2em; padding: 1.5em; bottom: 50px; right: 50px; background: red; color: #fff; border-radius: 2px; }' +
		'#byu-notice span { cursor: pointer; color: red; background: #fff; border-radius: 2px; padding: 0 5px; }' +

		'#byu-video-page-black { position: fixed; z-index: 999999; width: 20%; min-width: 200px; font-size: 1.1em; padding: 1em; bottom: 50px; left: 50px; background: red; color: #fff; border-radius: 2px; }' +

		'#byu { color: #A0A0A0; cursor: pointer; font-size: 22px; vertical-align: middle; } ' +

		'#byu-options { width: 500px; display: flex; flex-flow: row wrap; align-items: baseline; position: fixed; right: 70px; 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 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.wl { width: 40%; } ' +
		'#byu-options .byu-textarea.bl { width: 60%; } ' +
		'.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: auto; 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; width: .8em; } ' +
		'</style>');
	if (ver === 'old') $('head').append('<style>#byu-notice, #byu-options { font-size: 10px; } #byu { color: #808080; }</style>');

	// when the first page opened is 'watch', pause video if blacklisted
	if (pv && /\/watch/.test(window.location.href)) {
		$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 ytplayer !== 'undefined' && typeof ytplayer.config.args.author !== 'undefined') {
			pausevideo(ytplayer.config.args.author);
		} else { // for greasemonkey
			var waitUvideo = setInterval(function() {
				if ($(uVideo).text().trim().length) {
					clearInterval(waitUvideo);
					pausevideo($(uVideo).text());
				}
			}, 1000);
		}
	}

	// changes' ver
	var byuver = await GM.getValue('byuver', '1');
	if (byuver !== '2.4.1') {
		byuver = '2.4.1';
		GM.setValue('byuver', byuver);
		/*$('body').append('<div id="byu-notice">BLOCK YOUTUBE USERS [2.4]<br><br>- Changed again how to store users\' values<br>- Coming from a direct link to youtube, the video is automatically paused if the channel is blacklisted.<br><br><span>dismiss</span></div>');
		$('#byu-notice span').on('click', function() { $('#byu-notice').remove(); });*/
	}

	// 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 wl"><div>Whitelist</div><textarea rows="4" id="byu-whitelist-words">' + sWL + '</textarea></div>' +
		'<div class="byu-textarea bl"><div>Blacklist</div><textarea rows="4" id="byu-blacklist-words">' + sBL + '</textarea></div>' +
		'<div class="byu-sep">separator <input id="byu-sep-symbol" type="text" value="' + sep + '" maxlength="1" /></div>' +
		'<div class="byu-opt">click to add <input type="checkbox" name="clickadd" value="clickadd" checked="checked"></div>' +
		'<div class="byu-opt">pause blacklisted video <input type="checkbox" name="pausevideo" value="pausevideo" checked="checked"></div>' +
		'<div id="byu-suspend">suspend</div>' +
		'</div>');
	if (!add) $('input[name=clickadd]').prop('checked', false);
	if (!pv) $('input[name=pausevideo]').prop('checked', false);

	// for the button with the new layout, wait till the masthead is added
	var waiting = setInterval(function() {
		if (ver === 'old' || $('#buttons').length) {
			clearInterval(waiting);
			button();
		}
	}, 1000);

	function button() {
		if (ver === 'new') {
			$('#buttons').before('<div style="display: inline-block; position: relative; height: 28px; width: 30px"><span id="byu">B</span></div>');
			margintop = $('#container.ytd-masthead').height();
		} else {
			$('#upload-btn').before('<div style="display: inline-block; position: relative; height: 28px; width: 30px"><span id="byu">B</span></div>');
			margintop = $('#yt-masthead-container').height() + parseInt($('#yt-masthead-container').css('padding-top')) + parseInt($('#yt-masthead-container').css('padding-bottom'));
		}

		$('head').append('<style>#byu-options {top:' + margintop + 'px; }</style>');
	}


	/** BLACKLISTING **/

	// check if a username is whitelisted
	function ifWhite(u) {
		for(var z = 0; z < ytwhitelist.length; z++) {
			var w = ytwhitelist[z].trim().toLowerCase();
			if (w.length && u === w) {
				return true;
			}
		}
		return false;
	}

	// check if a username is blacklisted
	function ifMatch(u) {
		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
					b = b.split('*')[1];
					if (b.length && u.indexOf(b) !== -1) {
						return true;
					}
				} else { // exact match
					if (b.length && u === b) {
						return true;
					}
				}
			}
		}
		return false;
	}

	// do the thing
	function findMatch(s) {
		$(s).each(function() {
			var username = $(this).text().trim().toLowerCase();
			if (username) {
				// if the username is blacklisted
				if (!suspend && ifMatch(username)) {
					if (!$(this).closest(tClasses).attr('id', 'byu-is-black')) {
						$(this).closest(tClasses).attr('id', 'byu-is-black');
					}

				// add click
				} else if (add) {
					if (!$(this).siblings('.byu-add').length) {
						$('<span class="byu-add" data="' + username + '">[x]</span>').insertBefore($(this));
					} else if ($(this).siblings('.byu-add').attr('data') != username) {
						$(this).siblings('.byu-add').attr('data', username);
					}
				}
			}
		});
	}

	// the final search function
	function search() {
		// ! playlist?list=WL = Watch Later | feed/t... = History, Subscriptions
		if (!/\/(playlist\?list=WL|feed\/[^t]\w+)/.test(window.location.href)) {
			for (var i = 0; i < uClasses.length; i++) {
				findMatch(uClasses[i]);
			}
		}
	}
	search();


	/** EVENT LISTENERS **/

	// open and close options
	$('body').on('click', '#byu', function() {
		$('#byu-options').slideToggle();
		$(this).css('font-weight', $(this).css('font-weight') === '700' ? '400' : '700');
	});

	// save blacklist changes and research
	$saved = $('<span style="margin-right: 1em; font-size: .8em">saved</span>');
	$error = $('<span style="margin-right: 1em; font-size: .8em; color: red">ERROR! * NOT ALLOWED AS SEPARATOR</span>');
	$('#byu-saveblacklist').on('click', async function() {
		if ($('#byu-sep-symbol').val() === '*') {
			$(this).before($error);
			setTimeout(function() { $error.remove(); }, 4000);
		} else {
			// save new values
			await GM.setValue('savedblocks', $('#byu-blacklist-words').val());
			await GM.setValue('savedwhites', $('#byu-whitelist-words').val());
			await GM.setValue('sep', $('#byu-sep-symbol').val());
			// add notification
			$(this).before($saved);
			setTimeout(function() { $saved.remove(); }, 2000);
			// clear everything
			$('[id="byu-is-black"]').removeAttr('id');
			suspend = false;
			$('#byu-suspend').css('font-weight', '400');
			// research
			await getValues();
			search();
		}
	});

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

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

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

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

})(jQuery);