自动跳过 YouTube 广告

自动立即跳过 YouTube 广告。删除广告拦截器警告弹出窗口。非常轻量且高效。

目前为 2024-10-07 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name Auto Skip YouTube Ads
  3. // @name:vi Tự Động Bỏ Qua Quảng Cáo YouTube
  4. // @name:zh-CN 自动跳过 YouTube 广告
  5. // @name:zh-TW 自動跳過 YouTube 廣告
  6. // @name:ja YouTube 広告を自動スキップ
  7. // @name:ko YouTube 광고 자동 건너뛰기
  8. // @name:es Saltar Automáticamente Anuncios De YouTube
  9. // @name:ru Автоматический Пропуск Рекламы На YouTube
  10. // @name:id Lewati Otomatis Iklan YouTube
  11. // @name:hi YouTube विज्ञापन स्वचालित रूप से छोड़ें
  12. // @namespace https://github.com/tientq64/userscripts
  13. // @version 4.5.3
  14. // @description Automatically skip YouTube ads instantly. Remove the ad blocker warning pop-up. Very lightweight and efficient.
  15. // @description:vi Tự động bỏ qua quảng cáo YouTube ngay lập tức. Loại bỏ cửa sổ bật lên cảnh báo trình chặn quảng cáo. Rất nhẹ và hiệu quả.
  16. // @description:zh-CN 自动立即跳过 YouTube 广告。删除广告拦截器警告弹出窗口。非常轻量且高效。
  17. // @description:zh-TW 立即自動跳過 YouTube 廣告。刪除廣告攔截器警告彈出視窗。非常輕巧且高效。
  18. // @description:ja YouTube 広告を即座に自動的にスキップします。広告ブロッカーの警告ポップアップを削除します。非常に軽量で効率的です。
  19. // @description:ko YouTube 광고를 즉시 자동으로 건너뜁니다. 광고 차단 경고 팝업을 제거하세요. 매우 가볍고 효율적입니다.
  20. // @description:es Omita automáticamente los anuncios de YouTube al instante. Elimine la ventana emergente de advertencia del bloqueador de anuncios. Muy ligero y eficiente.
  21. // @description:ru Автоматически пропускайте рекламу YouTube мгновенно. Удалите всплывающее окно с предупреждением о блокировке рекламы. Очень легкий и эффективный.
  22. // @description:id Lewati iklan YouTube secara otomatis secara instan. Hapus pop-up peringatan pemblokir iklan. Sangat ringan dan efisien.
  23. // @description:hi YouTube विज्ञापनों को तुरंत स्वचालित रूप से छोड़ें। विज्ञापन अवरोधक चेतावनी पॉप-अप को हटाएँ। बहुत हल्का और कुशल।
  24. // @author tientq64
  25. // @icon https://cdn-icons-png.flaticon.com/64/2504/2504965.png
  26. // @match https://*.youtube.com/*
  27. // @grant GM_getValue
  28. // @grant GM_setValue
  29. // @grant GM_registerMenuCommand
  30. // @license MIT
  31. // @compatible firefox
  32. // @compatible chrome
  33. // @compatible opera
  34. // @compatible safari
  35. // @compatible edge
  36. // @noframes
  37. // @homepage https://github.com/tientq64/userscripts/tree/main/scripts/Auto-Skip-YouTube-Ads
  38. // ==/UserScript==
  39.  
  40. function skipAd() {
  41. video = null
  42. fineScrubbing = document.querySelector('.ytp-fine-scrubbing')
  43.  
  44. // Check if the current URL is a YouTube Shorts URL and exit the function if true
  45. if (window.location.pathname.startsWith('/shorts/')) return
  46.  
  47. const player = document.querySelector('#movie_player')
  48. let hasAd = false
  49.  
  50. if (player) {
  51. hasAd = player.classList.contains('ad-showing')
  52. video = player.querySelector('video.html5-main-video')
  53. }
  54.  
  55. if (hasAd) {
  56. const skipButton = document.querySelector(`
  57. .ytp-skip-ad-button,
  58. .ytp-ad-skip-button,
  59. .ytp-ad-skip-button-modern
  60. `)
  61. if (skipButton) {
  62. skipButton.click()
  63. skipButton.remove()
  64. } else if (video && video.src) {
  65. video.currentTime = 9999
  66. }
  67. }
  68.  
  69. if (video) {
  70. video.addEventListener('pause', handleVideoPause)
  71. video.addEventListener('mouseup', allowPauseVideo)
  72. }
  73.  
  74. const adBlockerWarningDialog = document.querySelector(
  75. 'tp-yt-paper-dialog:has(#feedback.ytd-enforcement-message-view-model)'
  76. )
  77. if (adBlockerWarningDialog) {
  78. adBlockerWarningDialog.remove()
  79. }
  80.  
  81. const adBlockerWarningInner = document.querySelector(
  82. '.yt-playability-error-supported-renderers'
  83. )
  84. if (adBlockerWarningInner) {
  85. if (config.allowedReloadPage) {
  86. adBlockerWarningInner.remove()
  87. location.reload()
  88. }
  89. }
  90.  
  91. const playButton = document.querySelector('button.ytp-play-button')
  92. if (playButton) {
  93. playButton.addEventListener('click', allowPauseVideo)
  94. }
  95.  
  96. const adShortVideos = document.querySelectorAll(
  97. 'ytd-reel-video-renderer:has(.ytd-ad-slot-renderer)'
  98. )
  99. for (const adShortVideo of adShortVideos) {
  100. adShortVideo.remove()
  101. }
  102. }
  103.  
  104. function allowPauseVideo() {
  105. pausedByUser = true
  106. window.clearTimeout(allowPauseVideoTimeoutId)
  107. allowPauseVideoTimeoutId = window.setTimeout(disallowPauseVideo, 500)
  108. }
  109.  
  110. function disallowPauseVideo() {
  111. pausedByUser = false
  112. window.clearTimeout(allowPauseVideoTimeoutId)
  113. }
  114.  
  115. function handleVideoPause() {
  116. if (pausedByUser) {
  117. disallowPauseVideo()
  118. return
  119. }
  120. if (document.hidden) return
  121. if (fineScrubbing && fineScrubbing.style.display !== 'none') return
  122. if (video) {
  123. if (video.duration - video.currentTime < 0.1) return
  124. video.play()
  125. }
  126. }
  127.  
  128. function handleGlobalKeyDownAndKeyUp(event) {
  129. if (document.activeElement?.matches('input, textarea, select')) return
  130. const code = event.code
  131. if (event.type === 'keydown') {
  132. if (code === 'KeyK' || code === 'MediaPlayPause') {
  133. allowPauseVideo()
  134. }
  135. } else {
  136. if (code === 'Space') {
  137. allowPauseVideo()
  138. }
  139. }
  140. }
  141.  
  142. function saveConfig() {
  143. GM_setValue('config', config)
  144. }
  145.  
  146. const defaultConfig = {
  147. allowedReloadPage: true
  148. }
  149.  
  150. const config = GM_getValue('config', defaultConfig)
  151. for (const key in defaultConfig) {
  152. if (config[key] == null) {
  153. config[key] = defaultConfig[key]
  154. }
  155. }
  156.  
  157. let video = null
  158. let fineScrubbing = null
  159. let pausedByUser = false
  160. let allowPauseVideoTimeoutId = 0
  161.  
  162. if (window.MutationObserver) {
  163. const observer = new MutationObserver(skipAd)
  164. observer.observe(document.body, {
  165. attributes: true,
  166. attributeFilter: ['class', 'src'],
  167. childList: true,
  168. subtree: true
  169. })
  170. } else {
  171. window.setInterval(skipAd, 500)
  172. }
  173. skipAd()
  174.  
  175. window.addEventListener('keydown', handleGlobalKeyDownAndKeyUp)
  176. window.addEventListener('keyup', handleGlobalKeyDownAndKeyUp)
  177.  
  178. const style = document.createElement('style')
  179. style.textContent = `
  180. #player-ads,
  181. #masthead-ad,
  182. #panels:has(ytd-ads-engagement-panel-content-renderer),
  183. ytd-ad-slot-renderer,
  184. ytd-rich-item-renderer:has(.ytd-ad-slot-renderer),
  185. ytd-reel-video-renderer:has(.ytd-ad-slot-renderer),
  186. tp-yt-paper-dialog:has(#feedback.ytd-enforcement-message-view-model),
  187. .ytp-suggested-action,
  188. .yt-mealbar-promo-renderer {
  189. display: none !important;
  190. }`
  191. document.head.appendChild(style)
  192.  
  193. {
  194. const title = 'Reload the page when there is no other way to skip ads'
  195. GM_registerMenuCommand(title, () => {
  196. config.allowedReloadPage = !config.allowedReloadPage
  197. alert(`${title}: ${config.allowedReloadPage ? 'ENABLED' : 'DISABLED'}`)
  198. })
  199. }

QingJ © 2025

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