YouTube 新标签页打开

功能:1.新标签页打开视频 2.屏蔽首页Shorts 3.屏蔽首页精选 4.主页内容刷新 5.非主页自动关闭 (所有功能可在右下角设置)

  1. // ==UserScript==
  2. // @name YouTube 新标签页打开
  3. // @namespace http://tampermonkey.net/
  4. // @version 1.3.3
  5. // @description 功能:1.新标签页打开视频 2.屏蔽首页Shorts 3.屏蔽首页精选 4.主页内容刷新 5.非主页自动关闭 (所有功能可在右下角设置)
  6. // @author YourName
  7. // @match *://www.youtube.com/*
  8. // @grant none
  9. // @license MIT
  10. // @icon https://www.google.com/s2/favicons?domain=youtube.com
  11. // ==/UserScript==
  12.  
  13. (function() {
  14. 'use strict';
  15.  
  16. // 从localStorage读取设置,默认全部开启
  17. var newTabEnabled = localStorage.getItem('yt_new_tab_enabled') !== null ?
  18. (localStorage.getItem('yt_new_tab_enabled') === 'true') : true;
  19. var shortsBlockingEnabled = localStorage.getItem('yt_shorts_blocking_enabled') !== null ?
  20. (localStorage.getItem('yt_shorts_blocking_enabled') === 'true') : true;
  21. var exploreBlockingEnabled = localStorage.getItem('yt_explore_blocking_enabled') !== null ?
  22. (localStorage.getItem('yt_explore_blocking_enabled') === 'true') : true;
  23.  
  24. // 监听链接点击事件
  25. document.addEventListener('click', function(e) {
  26. if (e.button !== 0 || e.ctrlKey || e.metaKey || e.shiftKey || e.altKey) return;
  27.  
  28. var anchor = e.target.closest('a');
  29. if (!anchor) return;
  30.  
  31. try {
  32. var url = new URL(anchor.href, window.location.origin);
  33.  
  34. // 处理视频链接
  35. if (newTabEnabled && url.pathname === '/watch' && url.searchParams.has('v')) {
  36. e.preventDefault();
  37. e.stopPropagation();
  38. window.open(anchor.href, '_blank');
  39. return;
  40. }
  41.  
  42. // 处理Shorts链接
  43. if (shortsBlockingEnabled && url.pathname.startsWith('/shorts')) {
  44. e.preventDefault();
  45. e.stopPropagation();
  46. return;
  47. }
  48. } catch (error) {
  49. return;
  50. }
  51. }, true);
  52.  
  53. // 创建悬浮控制容器
  54. var controlContainer = document.createElement('div');
  55. controlContainer.style.position = 'fixed';
  56. controlContainer.style.bottom = '10px';
  57. controlContainer.style.right = '10px';
  58. controlContainer.style.zIndex = '9999';
  59. controlContainer.style.display = 'flex';
  60. controlContainer.style.flexDirection = 'row';
  61. controlContainer.style.alignItems = 'flex-end';
  62.  
  63. // 创建设置菜单
  64. var menu = document.createElement('div');
  65. menu.style.display = 'none';
  66. menu.style.flexDirection = 'column';
  67. menu.style.backgroundColor = 'white';
  68. menu.style.border = '1px solid #ccc';
  69. menu.style.borderRadius = '4px';
  70. menu.style.padding = '10px';
  71. menu.style.position = 'fixed';
  72. menu.style.bottom = '60px';
  73. menu.style.right = '10px';
  74. menu.style.boxShadow = '0 2px 6px rgba(0,0,0,0.2)';
  75. menu.style.zIndex = '10000';
  76.  
  77. // 创建功能开关
  78. var newTabDiv = document.createElement('div');
  79. var newTabCheckbox = document.createElement('input');
  80. newTabCheckbox.type = 'checkbox';
  81. newTabCheckbox.checked = newTabEnabled;
  82. newTabCheckbox.id = 'newTabToggle';
  83. var newTabLabel = document.createElement('label');
  84. newTabLabel.htmlFor = 'newTabToggle';
  85. newTabLabel.textContent = ' 新标签页打开';
  86. newTabDiv.appendChild(newTabCheckbox);
  87. newTabDiv.appendChild(newTabLabel);
  88.  
  89. var shortsDiv = document.createElement('div');
  90. var shortsCheckbox = document.createElement('input');
  91. shortsCheckbox.type = 'checkbox';
  92. shortsCheckbox.checked = shortsBlockingEnabled;
  93. shortsCheckbox.id = 'shortsToggle';
  94. var shortsLabel = document.createElement('label');
  95. shortsLabel.htmlFor = 'shortsToggle';
  96. shortsLabel.textContent = ' Shorts 屏蔽';
  97. shortsDiv.appendChild(shortsCheckbox);
  98. shortsDiv.appendChild(shortsLabel);
  99.  
  100. var exploreDiv = document.createElement('div');
  101. var exploreCheckbox = document.createElement('input');
  102. exploreCheckbox.type = 'checkbox';
  103. exploreCheckbox.checked = exploreBlockingEnabled;
  104. exploreCheckbox.id = 'exploreToggle';
  105. var exploreLabel = document.createElement('label');
  106. exploreLabel.htmlFor = 'exploreToggle';
  107. exploreLabel.textContent = ' 精选屏蔽';
  108. exploreDiv.appendChild(exploreCheckbox);
  109. exploreDiv.appendChild(exploreLabel);
  110.  
  111. menu.appendChild(newTabDiv);
  112. menu.appendChild(shortsDiv);
  113. menu.appendChild(exploreDiv);
  114.  
  115. // 创建保存按钮
  116. var saveButton = document.createElement('button');
  117. saveButton.textContent = '保存';
  118. saveButton.style.marginTop = '10px';
  119. saveButton.style.padding = '5px 10px';
  120. saveButton.style.border = 'none';
  121. saveButton.style.backgroundColor = '#0073e6';
  122. saveButton.style.color = '#fff';
  123. saveButton.style.cursor = 'pointer';
  124. saveButton.style.borderRadius = '4px';
  125. menu.appendChild(saveButton);
  126. saveButton.addEventListener('click', function() {
  127. location.reload();
  128. });
  129.  
  130. // 绑定设置变更事件
  131. newTabCheckbox.addEventListener('change', function() {
  132. newTabEnabled = newTabCheckbox.checked;
  133. localStorage.setItem('yt_new_tab_enabled', newTabEnabled);
  134. });
  135.  
  136. shortsCheckbox.addEventListener('change', function() {
  137. shortsBlockingEnabled = shortsCheckbox.checked;
  138. localStorage.setItem('yt_shorts_blocking_enabled', shortsBlockingEnabled);
  139. });
  140.  
  141. exploreCheckbox.addEventListener('change', function() {
  142. exploreBlockingEnabled = exploreCheckbox.checked;
  143. localStorage.setItem('yt_explore_blocking_enabled', exploreBlockingEnabled);
  144. });
  145.  
  146. // 创建刷新/关闭按钮
  147. var refreshButton = document.createElement('div');
  148. refreshButton.textContent = '刷新内容';
  149. refreshButton.style.fontSize = '15px';
  150. refreshButton.style.fontWeight = 'bold';
  151. refreshButton.style.width = 'auto';
  152. refreshButton.style.height = '40px';
  153. refreshButton.style.borderRadius = '20px';
  154. refreshButton.style.backgroundColor = '#FF9EB5';
  155. refreshButton.style.color = 'white';
  156. refreshButton.style.display = 'flex';
  157. refreshButton.style.justifyContent = 'center';
  158. refreshButton.style.alignItems = 'center';
  159. refreshButton.style.cursor = 'pointer';
  160. refreshButton.style.userSelect = 'none';
  161. refreshButton.style.marginRight = '10px';
  162. refreshButton.style.boxShadow = '0 2px 6px rgba(0,0,0,0.1)';
  163. refreshButton.style.whiteSpace = 'nowrap';
  164. refreshButton.style.padding = '0 15px';
  165. refreshButton.style.transition = 'all 0.2s ease';
  166. refreshButton.style.position = 'relative';
  167. refreshButton.style.top = '0';
  168.  
  169. // 添加点击事件处理
  170. refreshButton.addEventListener('click', function() {
  171. if (window.location.pathname === '/') {
  172. const ytLogo = document.querySelector('#logo-icon');
  173. if (ytLogo) {
  174. ytLogo.click();
  175. }
  176. } else {
  177. window.close();
  178. }
  179. });
  180.  
  181. // 按钮悬停效果
  182. refreshButton.addEventListener('mouseenter', function() {
  183. this.style.boxShadow = '0 4px 12px rgba(0,0,0,0.2)';
  184. this.style.top = '-2px';
  185. });
  186. refreshButton.addEventListener('mouseleave', function() {
  187. this.style.boxShadow = '0 2px 6px rgba(0,0,0,0.1)';
  188. this.style.top = '0';
  189. });
  190.  
  191. // 更新按钮文字
  192. function updateButtonText() {
  193. refreshButton.textContent = window.location.pathname === '/' ? '刷新内容' : '关闭页面';
  194. }
  195. updateButtonText();
  196.  
  197. // 创建设置按钮
  198. var ball = document.createElement('div');
  199. ball.textContent = '⚙';
  200. ball.style.fontSize = '24px';
  201. ball.style.width = '40px';
  202. ball.style.height = '40px';
  203. ball.style.borderRadius = '50%';
  204. ball.style.backgroundColor = '#0073e6';
  205. ball.style.color = 'white';
  206. ball.style.display = 'flex';
  207. ball.style.justifyContent = 'center';
  208. ball.style.alignItems = 'center';
  209. ball.style.cursor = 'pointer';
  210. ball.style.userSelect = 'none';
  211. ball.style.lineHeight = '40px';
  212. ball.style.textAlign = 'center';
  213. ball.style.transition = 'all 0.2s ease';
  214. ball.style.position = 'relative';
  215. ball.style.top = '0';
  216. ball.style.boxShadow = '0 2px 6px rgba(0,0,0,0.1)';
  217.  
  218. ball.addEventListener('mouseenter', function() {
  219. this.style.boxShadow = '0 4px 12px rgba(0,0,0,0.2)';
  220. this.style.top = '-2px';
  221. });
  222. ball.addEventListener('mouseleave', function() {
  223. this.style.boxShadow = '0 2px 6px rgba(0,0,0,0.1)';
  224. this.style.top = '0';
  225. });
  226.  
  227. // 监听URL变化
  228. var urlObserver = new MutationObserver(updateButtonText);
  229. if (document.querySelector('title')) {
  230. urlObserver.observe(document.querySelector('title'), {
  231. subtree: true,
  232. characterData: true,
  233. childList: true
  234. });
  235. }
  236.  
  237. // 组装界面元素
  238. controlContainer.appendChild(refreshButton);
  239. controlContainer.appendChild(ball);
  240. document.body.appendChild(controlContainer);
  241. document.body.appendChild(menu);
  242.  
  243. // 全屏状态检测
  244. function handleFullscreen() {
  245. if (document.fullscreenElement) {
  246. controlContainer.style.display = 'none';
  247. menu.style.display = 'none';
  248. } else {
  249. controlContainer.style.display = 'flex';
  250. }
  251. }
  252.  
  253. // 监听全屏状态变化
  254. document.addEventListener('fullscreenchange', handleFullscreen);
  255. document.addEventListener('webkitfullscreenchange', handleFullscreen);
  256. document.addEventListener('mozfullscreenchange', handleFullscreen);
  257. document.addEventListener('MSFullscreenChange', handleFullscreen);
  258.  
  259. // 处理设置菜单的显示/隐藏
  260. ball.addEventListener('click', function(e) {
  261. e.stopPropagation();
  262. menu.style.display = (menu.style.display === 'none') ? 'flex' : 'none';
  263. });
  264.  
  265. // 点击外部关闭菜单
  266. document.addEventListener('click', function(e) {
  267. if (!menu.contains(e.target) && e.target !== ball) {
  268. menu.style.display = 'none';
  269. }
  270. });
  271.  
  272. // 移除Shorts和精选内容
  273. if (window.location.pathname === "/" && (shortsBlockingEnabled || exploreBlockingEnabled)) {
  274. function removeContent() {
  275. var shelves = document.querySelectorAll("ytd-rich-shelf-renderer, ytd-reel-shelf-renderer");
  276. shelves.forEach(function(shelf) {
  277. var titleElement = shelf.querySelector('h2');
  278. if (titleElement) {
  279. var titleText = titleElement.textContent.trim().toLowerCase();
  280. // 移除Shorts内容
  281. if (shortsBlockingEnabled && titleText.includes('shorts')) {
  282. shelf.remove();
  283. return;
  284. }
  285. // 移除精选内容
  286. if (exploreBlockingEnabled && (titleText.includes('精选') || titleText.includes('explore'))) {
  287. shelf.remove();
  288. return;
  289. }
  290. }
  291. // 备用检测:移除包含shorts链接的内容
  292. if (shortsBlockingEnabled && shelf.querySelector("a[href^='/shorts']")) {
  293. shelf.remove();
  294. }
  295. });
  296. }
  297.  
  298. removeContent();
  299. // 监听DOM变化,动态移除新增内容
  300. var observer = new MutationObserver(removeContent);
  301. observer.observe(document.body, { childList: true, subtree: true });
  302. }
  303. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址