Bigger Twitch

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

当前为 2014-08-18 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Bigger Twitch
  3. // @author Remos
  4. // @version 1.04
  5. // @description Removes the left nav bar entirely and stretches the twitch player to use as much space as possible.
  6. // @include http://*.twitch.tv/*
  7. // @include http://twitch.tv/*
  8. // @include http://*.twitch.tv/*/c/*
  9. // @include http://*.twitch.tv/*/b/*
  10. // @exclude http://www.twitch.tv/*/chat?popout=
  11. // @exclude http://www.twitch.tv/*/popout
  12. // @exclude http://www.twitch.tv/*/dashboard
  13. // @exclude http://www.twitch.tv/inbox*
  14. // @exclude http://www.twitch.tv/subscriptions*
  15. // @exclude http://store.twitch.tv
  16. // @exclude http://api.twitch.tv/*
  17. // @exclude https://api.twitch.tv/*
  18. // @require http://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js
  19. // @grant GM_addStyle
  20. // @grant GM_getValue
  21. // @grant GM_setValue
  22. // @copyright 2014, Remos
  23. // @run-at document-end
  24. // @icon https://monkeyguts.com/icon/288.png
  25. // @namespace 41229d0298d2565bc814c34b46b9158b
  26. // ==/UserScript==
  27.  
  28. // Icons used from http://icomoon.io/
  29.  
  30. GM_addStyle("#left_col.bigttv { visibility: hidden !important; }");
  31. GM_addStyle("#main_col.bigttv { margin-left: 0px !important; }");
  32. GM_addStyle(".ember-view.ember-chat.bigttv { min-width: 0px !important; }");
  33. 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; }");
  34. GM_addStyle(".bigttv-button.bigttv-contract { background-image: url(); }");
  35. 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(); }");
  36. GM_addStyle(".js-new-channel-ad { visibility: hidden !important; }");
  37.  
  38. GM_addStyle("#channel.bigttv { padding: 0 }");
  39.  
  40. GM_addStyle(".target-player { position: relative; }");
  41.  
  42. var chatWidth = GM_getValue('chatwidth', 340);
  43.  
  44. var buttonClickNum = 2;
  45. var buttonTitle = "Middle click to set chat width."; // Chrome doesn't like us using right click.
  46. if(navigator.userAgent.toLowerCase().indexOf('firefox') != -1) { // Firefox doesn't like letting us use middle click.
  47. buttonTitle = "Right click to set chat width.";
  48. buttonClickNum = 3;
  49. }
  50.  
  51. var biggerttvEnabled = false;
  52. var selfTriggered = false;
  53. var currentPlayer;
  54.  
  55. function updateSizes(player) {
  56. player = (typeof player === 'undefined') ? currentPlayer : player;
  57.  
  58. var columnVisible = $('#right_col').is(':visible');
  59. if(biggerttvEnabled) {
  60. var windowWidth = $(window).width();
  61. var windowHeight = $(window).height();
  62. var playerWidth = windowWidth;
  63.  
  64. if(columnVisible)
  65. playerWidth = windowWidth - chatWidth;
  66.  
  67. //$('#player, .archive_site_player_container, .target-player').attr('style', 'height: ' + windowHeight + 'px !important; width: ' + playerWidth + 'px !important;');
  68. $(player).attr('style', 'height: ' + windowHeight + 'px !important; width: ' + playerWidth + 'px !important;');
  69. $('[id$=flash-player], #archive_site_player_flash').attr('style', 'height: 100% !important; width: 100% !important; visibility: inherit !important');
  70. if(columnVisible)
  71. $('#main_col').css('margin-right', chatWidth);
  72. else
  73. $('#main_col').css('margin-right', 0);
  74. $('#main_col').width('auto');
  75.  
  76. $('#right_col').width(chatWidth);
  77. $('#right_col').width(chatWidth);
  78. $('#right_nav').parent().width(chatWidth);
  79. $('#right_nav').width(chatWidth - 19);
  80. $('#chat').width(chatWidth);
  81. $('.js-new-channel-ad').width(chatWidth);
  82.  
  83. currentPlayer = player;
  84. } else {
  85. currentPlayer = null;
  86.  
  87. $(player).attr('style', '');
  88.  
  89. var resetChatWidth = 340;
  90. if(!columnVisible && !$('#right_col').hasClass('ember-view'))
  91. resetChatWidth = 0;
  92. $('#right_col').width(resetChatWidth);
  93. $('#chat').width(340);
  94. $('#main_col').css('margin-right', '');
  95.  
  96. selfTriggered = true;
  97. window.dispatchEvent(new Event('resize'));
  98. selfTriggered = false;
  99. }
  100. }
  101.  
  102. function toggleBigger(enable, target) {
  103. if(typeof enable != 'boolean')
  104. enable = !biggerttvEnabled;
  105. biggerttvEnabled = enable;
  106.  
  107. $('#channel').toggleClass('bigttv', enable);
  108. $('#main_col').toggleClass('bigttv', enable);
  109. $('#left_col').toggleClass('bigttv', enable);
  110. $('.ember-view.ember-chat').toggleClass('bigttv', enable);
  111.  
  112. $(target).parent().toggleClass('bigttv', enable);
  113. //$('#player, .archive_site_player_container').toggleClass('bigttv', enable);
  114.  
  115. $('.bigttv-button').toggleClass('bigttv-contract', enable);
  116.  
  117. updateSizes($(target).parent());
  118. }
  119.  
  120. function createButton() {
  121. var button = $('<a title="' + buttonTitle + '" class="bigttv-button bigttv-expand"> </a>');
  122. button.unbind().click(function(e) {
  123. if(e.which == 1) {
  124. toggleBigger(null, e.target);
  125. return false;
  126. }
  127. }).mousedown(function(e) {
  128. if(e.which == buttonClickNum) {
  129. var newWidth = window.prompt("Enter new chat width:", GM_getValue('chatwidth', 340));
  130. if($.isNumeric(newWidth)) {
  131. chatWidth = parseInt(newWidth);
  132. GM_setValue('chatwidth', chatWidth);
  133. updateSizes($(e.target).parent());
  134. }
  135.  
  136. return false;
  137. }
  138. });
  139.  
  140. return button;
  141. }
  142.  
  143. var resizeTimer;
  144. $(window).resize(function() {
  145. if(selfTriggered)
  146. return;
  147. clearTimeout(resizeTimer);
  148. resizeTimer = setTimeout(updateSizes, (typeof BTTVLOADED === 'undefined') ? 0 : 1500);
  149. });
  150.  
  151. function addButton(items) {
  152. items.each(function(index) {
  153. if($(this).has('.bigttv-button').length == 0)
  154. $(this).append(createButton());
  155. });
  156. }
  157.  
  158. $('#player, .archive_site_player_container, .target-player').ready(function() {
  159. addButton($('#player, .archive_site_player_container, .target-player'))
  160. });
  161.  
  162. waitForKeyElements('.target-player', addButton);
  163.  
  164. unsafeWindow.bigbutton = createButton;
  165.  
  166. // ======================================
  167.  
  168. function waitForKeyElements (
  169. selectorTxt, /* Required: The jQuery selector string that
  170. specifies the desired element(s).
  171. */
  172. actionFunction, /* Required: The code to run when elements are
  173. found. It is passed a jNode to the matched
  174. element.
  175. */
  176. bWaitOnce, /* Optional: If false, will continue to scan for
  177. new elements even after the first match is
  178. found.
  179. */
  180. iframeSelector /* Optional: If set, identifies the iframe to
  181. search.
  182. */
  183. ) {
  184. var targetNodes, btargetsFound;
  185. if (typeof iframeSelector == "undefined")
  186. targetNodes = $(selectorTxt);
  187. else
  188. targetNodes = $(iframeSelector).contents ()
  189. .find (selectorTxt);
  190. if (targetNodes && targetNodes.length > 0) {
  191. btargetsFound = true;
  192. /*--- Found target node(s). Go through each and act if they
  193. are new.
  194. */
  195. targetNodes.each ( function () {
  196. var jThis = $(this);
  197. var alreadyFound = jThis.data ('alreadyFound') || false;
  198. if (!alreadyFound) {
  199. //--- Call the payload function.
  200. var cancelFound = actionFunction (jThis);
  201. if (cancelFound)
  202. btargetsFound = false;
  203. else
  204. jThis.data ('alreadyFound', true);
  205. }
  206. } );
  207. }
  208. else {
  209. btargetsFound = false;
  210. }
  211. //--- Get the timer-control variable for this selector.
  212. var controlObj = waitForKeyElements.controlObj || {};
  213. var controlKey = selectorTxt.replace (/[^\w]/g, "_");
  214. var timeControl = controlObj [controlKey];
  215. //--- Now set or clear the timer as appropriate.
  216. if (btargetsFound && bWaitOnce && timeControl) {
  217. //--- The only condition where we need to clear the timer.
  218. clearInterval (timeControl);
  219. delete controlObj [controlKey]
  220. }
  221. else {
  222. //--- Set a timer, if needed.
  223. if ( ! timeControl) {
  224. timeControl = setInterval ( function () {
  225. waitForKeyElements ( selectorTxt,
  226. actionFunction,
  227. bWaitOnce,
  228. iframeSelector
  229. );
  230. },
  231. 300
  232. );
  233. controlObj [controlKey] = timeControl;
  234. }
  235. }
  236. waitForKeyElements.controlObj = controlObj;
  237. }