YouTube 小助手

YouTube 1.视频循环播放 2.截图下载 3.主题进度条

当前为 2025-03-14 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name YouTube Helper
  3. // @name:zh-CN YouTube 小助手
  4. // @description YouTube 1.Loop playback of YouTube videos 2.screenshot download 3.themed progress bar.
  5. // @description:zh-CN YouTube 1.视频循环播放 2.截图下载 3.主题进度条
  6. // @author Carokahn,bernzrdo,FunnyMonkey,人民的勤务员 <china.qinwuyuan@gmail.com>
  7. // @namespace https://github.com/ChinaGodMan/UserScripts
  8. // @supportURL https://github.com/ChinaGodMan/UserScripts/issues
  9. // @homepageURL https://github.com/ChinaGodMan/UserScripts
  10. // @license MIT
  11. // @icon https://www.youtube.com/s/desktop/ee47b5e0/img/logos/favicon_144x144.png
  12. // @match https://www.youtube.com/*
  13. // @match https://m.youtube.com/*
  14. // @compatible chrome
  15. // @compatible firefox
  16. // @compatible edge
  17. // @compatible opera
  18. // @compatible safari
  19. // @compatible kiwi
  20. // @compatible qq
  21. // @compatible via
  22. // @compatible brave
  23. // @version 2025.03.15.0436
  24. // @grant GM_addStyle
  25. // @created 2025-03-14 20:36:01
  26. // @modified 2025-03-14 20:36:01
  27. // ==/UserScript==
  28. /**
  29. * File: youtube-helper.user.js
  30. * Project: UserScripts
  31. * File Created: 2025/03/15,Saturday 04:36:02
  32. * Author: 人民的勤务员@ChinaGodMan (china.qinwuyuan@gmail.com)
  33. * -----
  34. * Last Modified: 2025/03/15,Saturday 05:57:23
  35. * Modified By: 人民的勤务员@ChinaGodMan (china.qinwuyuan@gmail.com)
  36. * -----
  37. * License: MIT License
  38. * Copyright © 2024 - 2025 ChinaGodMan,Inc
  39. */
  40. const directDownload = true
  41. const infiniteLool = true
  42.  
  43.  
  44.  
  45.  
  46.  
  47. const loopVideo = () => {
  48. const video = document.querySelector('video')
  49. if (video && !video.loop) {
  50. video.loop = true
  51. }
  52. }
  53. const ThemeProgressbar = () => {
  54. const css_248z = '.html5-play-progress,.ytp-play-progress{background:url("") repeat-x!important;background:linear-gradient(180deg,red 0,red 16.5%,#f90 0,#f90 33%,#ff0 0,#ff0 50%,#3f0 0,#3f0 66%,#09f 0,#09f 83.5%,#63f 0,#63f)!important;background:-webkit-linear-gradient(top,red,red 16.5%,#f90 0,#f90 33%,#ff0 0,#ff0 50%,#3f0 0,#3f0 66%,#09f 0,#09f 83.5%,#63f 0,#63f)!important;background:-moz-linear-gradient(top,red 0,red 16.5%,#f90 16.5%,#f90 33%,#ff0 33%,#ff0 50%,#3f0 50%,#3f0 66%,#09f 66%,#09f 83.5%,#63f 83.5%,#63f 100%)!important}.html5-load-progress,.ytp-load-progress{background:url("")!important}.html5-scrubber-button,.ytp-scrubber-button{background:url("")!important;border:none!important;height:21px!important;margin-left:-18px!important;margin-top:0!important;transform:scale(.8);-webkit-transform:scale(.8);-moz-transform:scale(.8);-ms-transform:scale(.8);width:34px!important}.ytp-progress-bar-container:hover .ytp-load-progress,.ytp-progress-bar-container:hover .ytp-scrubber-button{image-rendering:pixelated}.html5-progress-bar-container,.ytp-progress-bar-container{height:12px!important}.html5-progress-bar,.ytp-progress-bar{margin-top:12px!important}.html5-progress-list,.video-ads .html5-progress-list.html5-ad-progress-list,.video-ads .ytp-progress-list.ytp-ad-progress-list,.ytp-progress-list{height:12px!important}.ytp-volume-slider-track{background:#0c4177!important}'
  55. GM_addStyle(css_248z)
  56. }
  57.  
  58. let escapeHTMLPolicy = 'trustedTypes' in window
  59. ? trustedTypes.createPolicy('forceInner', { createHTML: html => html })
  60. : { createHTML: html => html }
  61. function screenBtnUpdate() {
  62. let $miniplayerBtn = document.querySelector('button.ytp-miniplayer-button')
  63. if ($miniplayerBtn && !document.getElementById('ytp-screenshot-button')) {
  64. const $btn = document.createElement('button')
  65. $btn.id = 'ytp-screenshot-button'
  66. $btn.classList.add('ytp-screenshot-button', 'ytp-button')
  67. $btn.dataset.priority = '5'
  68. $btn.dataset.tooltipTargetId = 'ytp-screenshot-button'
  69. $btn.dataset.titleNoTooltip = 'Screenshot'
  70. $btn.ariaLabel = 'Screenshot'
  71. $btn.title = 'Screenshot'
  72. $btn.innerHTML = escapeHTMLPolicy.createHTML(`<svg height="100%" version="1.1" viewBox="-300 -1260 1560 1560" width="100%">
  73. <use class="ytp-svg-shadow" xlink:href="#ytp-id-screenshot-svg"></use>
  74. <path
  75. d="M200-120q-33 0-56.5-23.5T120-200v-560q0-33 23.5-56.5T200-840h560q33 0 56.5 23.5T840-760v560q0 33-23.5 56.5T760-120H200Zm0-80h560v-560H200v560Zm40-80h480L570-480 450-320l-90-120-120 160Zm-40 80v-560 560Z"
  76. fill="#fff" id="ytp-id-screenshot-svg"></path>
  77. </svg>`)
  78. $btn.addEventListener('click', screenshot)
  79.  
  80. insertBefore($btn, $miniplayerBtn)
  81. }
  82.  
  83. requestAnimationFrame(screenBtnUpdate)
  84. }
  85. function insertBefore($element, $sibling) {
  86. $sibling.parentElement.insertBefore($element, $sibling)
  87. }
  88. function screenshot() {
  89.  
  90. const $video = document.querySelector('#player video')
  91. if (!$video) {
  92. console.error('No video found to screenshot!')
  93. return
  94. }
  95.  
  96. let wasPlaying = !$video.paused
  97. if (wasPlaying) $video.pause()
  98.  
  99. const $canvas = document.createElement('canvas')
  100. $canvas.width = $video.videoWidth
  101. $canvas.height = $video.videoHeight
  102.  
  103. let oldCss = $video.style.cssText
  104. $video.style.width = $video.videoWidth + 'px'
  105. $video.style.height = $video.videoHeight + 'px'
  106.  
  107. const ctx = $canvas.getContext('2d')
  108. ctx.drawImage($video, 0, 0, $video.videoWidth, $video.videoHeight)
  109.  
  110. $canvas.toBlob(blob => {
  111.  
  112. if (directDownload) {
  113. const a = document.createElement('a')
  114. a.href = URL.createObjectURL(blob)
  115. a.download = `${getFileName()}.png`
  116. a.click()
  117. } else {
  118. const item = new ClipboardItem({ 'image/png': blob })
  119. navigator.clipboard.write([item])
  120. }
  121.  
  122. $video.style.cssText = oldCss
  123. $canvas.remove()
  124. if (wasPlaying) $video.play()
  125.  
  126. })
  127.  
  128. }
  129. function getFileName() {
  130. const safeFileName = document.title.replace(/[\\/:*?"<>|]/g, '').replace(' - YouTube', '')
  131. return safeFileName
  132. }
  133. if (infiniteLool) {
  134. const observer = new MutationObserver(loopVideo)
  135. observer.observe(document.body, { childList: true, subtree: true })
  136. }
  137.  
  138. requestAnimationFrame(screenBtnUpdate)
  139. ThemeProgressbar()