Bigger Twitch

Removes the left nav bar entirely and stretches the twitch player to use as much space as possible.

Mint 2014.08.18.. Lásd a legutóbbi verzió

// ==UserScript==
// @name           Bigger Twitch
// @author         Remos
// @version        1.03.2
// @description    Removes the left nav bar entirely and stretches the twitch player to use as much space as possible.
// @include        http://*.twitch.tv/*
// @include        http://twitch.tv/*
// @include        http://*.twitch.tv/*/c/*
// @include        http://*.twitch.tv/*/b/*
// @exclude        http://www.twitch.tv/*/chat?popout=
// @exclude        http://www.twitch.tv/*/popout
// @exclude        http://www.twitch.tv/*/dashboard
// @exclude        http://www.twitch.tv/inbox*
// @exclude        http://www.twitch.tv/subscriptions*
// @exclude        http://store.twitch.tv
// @exclude        http://api.twitch.tv/*
// @exclude        https://api.twitch.tv/*
// @require        http://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js
// @grant          GM_addStyle
// @grant          GM_getValue
// @grant          GM_setValue
// @copyright      2014, Remos
// @run-at         document-end
// @icon           https://monkeyguts.com/icon/288.png
// @namespace   41229d0298d2565bc814c34b46b9158b
// ==/UserScript==

// Icons used from http://icomoon.io/

GM_addStyle("#left_col.bigttv  { visibility: hidden !important; }");
GM_addStyle("#main_col.bigttv  { margin-left: 0px !important; }");
GM_addStyle(".ember-view.ember-chat.bigttv { min-width: 0px !important; }");
GM_addStyle(".player-column #player.bigttv, .archive_site_player_container.bigttv, .target-player.bigttv { position: fixed !important; top: 0px !important; left: 0px !important; z-index: 99999; }");
GM_addStyle(".bigttv-button.bigttv-contract { background-image: url(); }");
GM_addStyle(".bigttv-button { cursor: pointer; position: absolute; bottom: 0px; left: 160px; padding: 3px; width: 21px; height: 21px; background-repeat: no-repeat; background-position: 3px; background-size: 18px; background-image: url(); }");
GM_addStyle(".js-new-channel-ad { visibility: hidden !important; }");

GM_addStyle("#channel.bigttv { padding: 0 }");

GM_addStyle(".target-player { position: relative; }");

var chatWidth = GM_getValue('chatwidth', 340);

var buttonClickNum = 2;
var buttonTitle = "Middle click to set chat width."; // Chrome doesn't like us using right click.
if(navigator.userAgent.toLowerCase().indexOf('firefox') != -1) { // Firefox doesn't like letting us use middle click.
	buttonTitle = "Right click to set chat width.";
	buttonClickNum = 3;
}

var biggerttvEnabled = false;
var selfTriggered = false;
var currentPlayer;

function updateSizes(player) {
	player = (typeof player === 'undefined') ? currentPlayer : player;

	var columnVisible = $('#right_col').is(':visible');
	if(biggerttvEnabled) {
		var windowWidth = $(window).width();
		var windowHeight = $(window).height();
		var playerWidth = windowWidth;

		if(columnVisible)
			playerWidth = windowWidth - chatWidth;

		//$('#player, .archive_site_player_container, .target-player').attr('style', 'height: ' + windowHeight + 'px !important; width: ' + playerWidth + 'px !important;');
		$(player).attr('style', 'height: ' + windowHeight + 'px !important; width: ' + playerWidth + 'px !important;');
		$('[id$=flash-player], #archive_site_player_flash').attr('style', 'height: 100% !important; width: 100% !important; visibility: inherit !important');
		if(columnVisible)
			$('#main_col').css('margin-right', chatWidth);
		else
			$('#main_col').css('margin-right', 0);
		$('#main_col').width('auto');

		$('#right_col').width(chatWidth);
		$('#right_col').width(chatWidth);
		$('#right_nav').parent().width(chatWidth);
		$('#right_nav').width(chatWidth - 19);
		$('#chat').width(chatWidth);
		$('.js-new-channel-ad').width(chatWidth);

		currentPlayer = player;
	} else {
		currentPlayer = null;

		$(player).attr('style', '');

		var resetChatWidth = 340;
		if(!columnVisible && !$('#right_col').hasClass('ember-view'))
			resetChatWidth = 0;
		$('#right_col').width(resetChatWidth);
		$('#chat').width(340);
		$('#main_col').css('margin-right', '');

		selfTriggered = true;
		window.dispatchEvent(new Event('resize'));
		selfTriggered = false;
	}
}

function toggleBigger(enable, target) {
	if(typeof enable != 'boolean')
		enable = !biggerttvEnabled;
	biggerttvEnabled = enable;

	$('#channel').toggleClass('bigttv', enable);
	$('#main_col').toggleClass('bigttv', enable);
	$('#left_col').toggleClass('bigttv', enable);
	$('.ember-view.ember-chat').toggleClass('bigttv', enable);

	$(target).parent().toggleClass('bigttv', enable);
	//$('#player, .archive_site_player_container').toggleClass('bigttv', enable);

	$('.bigttv-button').toggleClass('bigttv-contract', enable);

	updateSizes($(target).parent());
}

function createButton() {
	var button = $('<a title="' + buttonTitle + '" class="bigttv-button bigttv-expand"> </a>');
	button.unbind().click(function(e) {
		if(e.which == 1) {
			toggleBigger(null, e.target);
			return false;
		}
	}).mousedown(function(e) {
		if(e.which == buttonClickNum) {
			var newWidth = window.prompt("Enter new chat width:", GM_getValue('chatwidth', 340));
			if($.isNumeric(newWidth)) {
				chatWidth = parseInt(newWidth);
				GM_setValue('chatwidth', chatWidth);
				updateSizes($(e.target).parent());
			}

			return false;
		}
	});

	return button;
}

var resizeTimer;
$(window).resize(function() {
	if(selfTriggered)
		return;
	clearTimeout(resizeTimer);
	resizeTimer = setTimeout(updateSizes, (typeof BTTVLOADED === 'undefined') ? 0 : 1500);
});

function addButton(items) {
	items.each(function(index) {
		if($(this).has('.bigttv-button').length == 0)
			$(this).append(createButton());
	});
}

$('#player, .archive_site_player_container, .target-player').ready(function() {
	addButton($('#player, .archive_site_player_container, .target-player'))
});

waitForKeyElements('.target-player', addButton);

unsafeWindow.bigbutton = createButton;

// ======================================

function waitForKeyElements (
	selectorTxt,    /* Required: The jQuery selector string that
						specifies the desired element(s).
					*/
	actionFunction, /* Required: The code to run when elements are
						found. It is passed a jNode to the matched
						element.
					*/
	bWaitOnce,      /* Optional: If false, will continue to scan for
						new elements even after the first match is
						found.
					*/
	iframeSelector  /* Optional: If set, identifies the iframe to
						search.
					*/
) {
	var targetNodes, btargetsFound;
 
	if (typeof iframeSelector == "undefined")
		targetNodes     = $(selectorTxt);
	else
		targetNodes     = $(iframeSelector).contents ()
										   .find (selectorTxt);
 
	if (targetNodes  &&  targetNodes.length > 0) {
		btargetsFound   = true;
		/*--- Found target node(s).  Go through each and act if they
			are new.
		*/
		targetNodes.each ( function () {
			var jThis        = $(this);
			var alreadyFound = jThis.data ('alreadyFound')  ||  false;
 
			if (!alreadyFound) {
				//--- Call the payload function.
				var cancelFound     = actionFunction (jThis);
				if (cancelFound)
					btargetsFound   = false;
				else
					jThis.data ('alreadyFound', true);
			}
		} );
	}
	else {
		btargetsFound   = false;
	}
 
	//--- Get the timer-control variable for this selector.
	var controlObj      = waitForKeyElements.controlObj  ||  {};
	var controlKey      = selectorTxt.replace (/[^\w]/g, "_");
	var timeControl     = controlObj [controlKey];
 
	//--- Now set or clear the timer as appropriate.
	if (btargetsFound  &&  bWaitOnce  &&  timeControl) {
		//--- The only condition where we need to clear the timer.
		clearInterval (timeControl);
		delete controlObj [controlKey]
	}
	else {
		//--- Set a timer, if needed.
		if ( ! timeControl) {
			timeControl = setInterval ( function () {
					waitForKeyElements (    selectorTxt,
											actionFunction,
											bWaitOnce,
											iframeSelector
										);
				},
				300
			);
			controlObj [controlKey] = timeControl;
		}
	}
	waitForKeyElements.controlObj   = controlObj;
}