优酷全屏模式

YouTube画面自动转换成100高、100宽的画面。

当前为 2024-12-26 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name:ko 유튜브 풀스크린
  3. // @name Youtube Fullscreen Mode
  4. // @name:ru Youtube Полный режим
  5. // @name:ja Youtubeフルスクリーンモードの
  6. // @name:zh-CN 优酷全屏模式
  7. // @name:zh-TW 優酷全屏模式
  8.  
  9. // @description:ko 유튜브 화면을 자동으로 꽉 찬 화면으로 바꿉니다.
  10. // @description Automatically switch YouTube screens to 100 height and 100 width screens.
  11. // @description:ru Автоматическое переключение экрана на YouTube на экран высотой 100 и шириной 100.
  12. // @description:ja ユーチューブ画面を高さ100、広さ100画面に自動転換。
  13. // @description:zh-CN YouTube画面自动转换成100高、100宽的画面。
  14. // @description:zh-TW YouTube畫面自動轉換成100高、100寬的畫面。
  15.  
  16. // @namespace https://ndaesik.tistory.com/
  17. // @version 2024.12.26.2245
  18. // @author ndaesik
  19. // @icon https://lh3.googleusercontent.com/iLZyxGK7l1343U4E7eAfgKbRWW6qhzCJq-Z92M60JzCMntFyaFF2GUQVRxPhfGcy6qRISLjHv4fX1vtq0TZkZMAzBjM
  20. // @match *://*.youtube.com/*
  21. // @grant window.focus
  22. // ==/UserScript==
  23.  
  24. const suggestBoxToDarkCSS = document.createElement('style');
  25. suggestBoxToDarkCSS.innerText = `
  26. [dark] {color-scheme: dark;}`.replaceAll(';','!important;')
  27.  
  28. const fullscreenVideoCSS = document.createElement('style');
  29. fullscreenVideoCSS.innerText = `
  30. ytd-app:not([guide-persistent-and-visible]) [theater] #player video,
  31. :is(ytd-watch-flexy[theater],ytd-watch-flexy[fullscreen]) #full-bleed-container {
  32. height: 100vh; max-height: 100vh; min-height: 100vh;}`.replaceAll(';','!important;')
  33.  
  34. const autoHideTopCSS = document.createElement('style');
  35. autoHideTopCSS.innerText = `
  36. #masthead-container.ytd-app:hover, #masthead-container.ytd-app:focus-within {width:100%;}
  37. #masthead-container.ytd-app,
  38. #masthead-container.ytd-app:not(:hover):not(:focus-within) {width:calc(50% - 150px);}
  39. #masthead-container.ytd-app:not(:hover):not(:focus-within) {transition:width 0.4s ease-out 0.4s;}
  40. ytd-app:not([guide-persistent-and-visible]) :is(#masthead-container ytd-masthead, #masthead-container.ytd-app::after) {transform: translateY(-56px); transition: transform .1s .3s ease-out;}
  41. ytd-app:not([guide-persistent-and-visible]) :is(#masthead-container:hover ytd-masthead, #masthead-container:hover.ytd-app::after, #masthead-container:focus-within ytd-masthead) {transform: translateY(0px);}
  42. ytd-app:not([guide-persistent-and-visible]) ytd-page-manager {margin-top: 0;}`.replaceAll(';','!important;')
  43. autoHideTopCSS.className = "autoHideTopCSS";
  44.  
  45. const $ = {
  46. elements: {
  47. ytdApp: null,
  48. player: null,
  49. chatFrame: null
  50. },
  51. update() {
  52. this.elements.ytdApp = document.querySelector('ytd-app');
  53. this.elements.player = document.querySelector('#ytd-player');
  54. this.elements.chatFrame = document.querySelector('ytd-live-chat-frame');
  55. }
  56. };
  57.  
  58. const isWatchPage = () => document.URL.includes('watch');
  59.  
  60. const isTheaterMode = () => {
  61. $.update();
  62. const { ytdApp, player, chatFrame } = $.elements;
  63. return ytdApp && player && isWatchPage() &&
  64. (window.innerWidth - ytdApp.offsetWidth +
  65. player.offsetWidth +
  66. (chatFrame && !chatFrame.attributes.collapsed ? chatFrame.offsetWidth : 0))
  67. === window.innerWidth;
  68. };
  69.  
  70. const shouldShowAutoHideCSS = () => {
  71. const scrollPosition = window.scrollY;
  72. const viewportHeight = window.innerHeight + 56;
  73. return isWatchPage() &&
  74. isTheaterMode() &&
  75. scrollPosition <= viewportHeight;
  76. };
  77.  
  78. const updateAutoHideCSS = () => {
  79. const existingCSS = document.querySelector('.autoHideTopCSS');
  80. if (shouldShowAutoHideCSS()) {
  81. !existingCSS && document.head.appendChild(autoHideTopCSS);
  82. } else {
  83. existingCSS?.remove();
  84. }
  85. };
  86.  
  87. const alwaysTheaterMode = () => {
  88. const interval = setInterval(() =>
  89. isTheaterMode()
  90. ? clearInterval(interval)
  91. : document.querySelectorAll('.ytp-size-button')?.forEach(e => e.click()),
  92. 100
  93. );
  94. setTimeout(() => clearInterval(interval), 10000);
  95. };
  96.  
  97. ['yt-navigate-finish', 'load', 'unload', 'locationchange'].forEach(event =>
  98. window.addEventListener(event, () => {
  99. document.head.appendChild(suggestBoxToDarkCSS);
  100. document.head.appendChild(fullscreenVideoCSS);
  101. alwaysTheaterMode();
  102. window.scrollTo(0, 0);
  103. updateAutoHideCSS();
  104. })
  105. );
  106.  
  107. window.addEventListener('click', () => {
  108. setTimeout(updateAutoHideCSS, 100);
  109. });
  110.  
  111. window.addEventListener('scroll', () => {
  112. requestAnimationFrame(updateAutoHideCSS);
  113. });