Greasy Fork is available in English.

YouTube Mute and Skip Ads

Mutes, blurs and skips ads on YouTube. Speeds up ad playback. Clicks "yes" on "are you there?" on YouTube Music.

  1. // ==UserScript==
  2. // @name YouTube Mute and Skip Ads
  3. // @namespace https://github.com/ion1/userscripts
  4. // @version 0.0.28
  5. // @author ion
  6. // @description Mutes, blurs and skips ads on YouTube. Speeds up ad playback. Clicks "yes" on "are you there?" on YouTube Music.
  7. // @license MIT
  8. // @icon https://www.google.com/s2/favicons?sz=64&domain=youtube.com
  9. // @homepage https://github.com/ion1/userscripts/tree/master/packages/youtube-mute-skip-ads
  10. // @homepageURL https://github.com/ion1/userscripts/tree/master/packages/youtube-mute-skip-ads
  11. // @match *://www.youtube.com/*
  12. // @match *://music.youtube.com/*
  13. // @grant GM_addStyle
  14. // @run-at document-body
  15. // ==/UserScript==
  16.  
  17. (n=>{if(typeof GM_addStyle=="function"){GM_addStyle(n);return}const e=document.createElement("style");e.textContent=n,document.head.append(e)})(` /* Keep these in sync with the watchers. */
  18. #movie_player
  19. :is(.ytp-ad-skip-button, .ytp-ad-skip-button-modern, .ytp-skip-ad-button) {
  20. anchor-name: --youtube-mute-skip-ads-unclickable-button;
  21. }
  22.  
  23. body:has(
  24. #movie_player
  25. :is(
  26. .ytp-ad-skip-button,
  27. .ytp-ad-skip-button-modern,
  28. .ytp-skip-ad-button
  29. ):not([style*="display: none"], [aria-hidden="true"])
  30. )::after {
  31. content: "\u{1D606}\u{1D5FC}\u{1D602}\u{1D601}\u{1D602}\u{1D5EF}\u{1D5F2}-\u{1D5FA}\u{1D602}\u{1D601}\u{1D5F2}-\u{1D600}\u{1D5F8}\u{1D5F6}\u{1D5FD}-\u{1D5EE}\u{1D5F1}\u{1D600}\\A\\A"
  32. "Unfortunately, YouTube has started to block automated clicks based on isTrusted being false.\\A\\A"
  33. "Please click on the skip button manually.";
  34. white-space: pre-line;
  35. pointer-events: none;
  36. z-index: 9999;
  37. position: fixed;
  38. position-anchor: --youtube-mute-skip-ads-unclickable-button;
  39. padding: 1.5em;
  40. border-radius: 1.5em;
  41. margin-bottom: 1em;
  42. bottom: anchor(--youtube-mute-skip-ads-unclickable-button top);
  43. right: anchor(--youtube-mute-skip-ads-unclickable-button right);
  44. max-width: 25em;
  45. font-size: 1.4rem;
  46. line-height: 2rem;
  47. font-weight: 400;
  48. color: rgb(240 240 240);
  49. background-color: rgb(0 0 0 / 0.7);
  50. backdrop-filter: blur(10px);
  51. animation: fade-in 3s linear;
  52. }
  53.  
  54. @keyframes fade-in {
  55. 0% {
  56. opacity: 0;
  57. }
  58. 67% {
  59. opacity: 0;
  60. }
  61. 100% {
  62. opacity: 1;
  63. }
  64. }
  65.  
  66. #movie_player.ad-showing video {
  67. filter: blur(100px) opacity(0.25) grayscale(0.5);
  68. }
  69.  
  70. #movie_player.ad-showing .ytp-title,
  71. #movie_player.ad-showing .ytp-title-channel,
  72. .ytp-visit-advertiser-link,
  73. .ytp-ad-visit-advertiser-button,
  74. ytmusic-app:has(#movie_player.ad-showing)
  75. ytmusic-player-bar
  76. :is(.title, .subtitle) {
  77. filter: blur(4px) opacity(0.5) grayscale(0.5);
  78. transition: 0.05s filter linear;
  79. }
  80.  
  81. :is(#movie_player.ad-showing .ytp-title,#movie_player.ad-showing .ytp-title-channel,.ytp-visit-advertiser-link,.ytp-ad-visit-advertiser-button,ytmusic-app:has(#movie_player.ad-showing) ytmusic-player-bar :is(.title,.subtitle)):is(:hover,:focus-within) {
  82. filter: none;
  83. }
  84.  
  85. /* These popups are showing up on top of the video with a hidden dismiss button
  86. * since 2024-09-25.
  87. */
  88. .ytp-suggested-action-badge {
  89. visibility: hidden !important;
  90. }
  91.  
  92. #movie_player.ad-showing .caption-window,
  93. .ytp-ad-player-overlay-flyout-cta,
  94. .ytp-ad-player-overlay-layout__player-card-container, /* Seen since 2024-04-06. */
  95. .ytp-ad-action-interstitial-slot, /* Added on 2024-08-25. */
  96. ytd-action-companion-ad-renderer,
  97. ytd-display-ad-renderer,
  98. ytd-ad-slot-renderer,
  99. ytd-promoted-sparkles-web-renderer,
  100. ytd-player-legacy-desktop-watch-ads-renderer,
  101. ytd-engagement-panel-section-list-renderer[target-id="engagement-panel-ads"],
  102. ytd-merch-shelf-renderer {
  103. filter: blur(10px) opacity(0.25) grayscale(0.5);
  104. transition: 0.05s filter linear;
  105. }
  106.  
  107. :is(#movie_player.ad-showing .caption-window,.ytp-ad-player-overlay-flyout-cta,.ytp-ad-player-overlay-layout__player-card-container,.ytp-ad-action-interstitial-slot,ytd-action-companion-ad-renderer,ytd-display-ad-renderer,ytd-ad-slot-renderer,ytd-promoted-sparkles-web-renderer,ytd-player-legacy-desktop-watch-ads-renderer,ytd-engagement-panel-section-list-renderer[target-id="engagement-panel-ads"],ytd-merch-shelf-renderer):is(:hover,:focus-within) {
  108. filter: none;
  109. }
  110.  
  111. .ytp-ad-action-interstitial-background-container /* Added on 2024-08-25. */ {
  112. /* An image ad in place of the video. */
  113. filter: blur(10px) opacity(0.25) grayscale(0.5);
  114. } `);
  115.  
  116. (function () {
  117. 'use strict';
  118.  
  119. const logPrefix = "youtube-mute-skip-ads:";
  120. class Watcher {
  121. name;
  122. element;
  123. #onCreatedCallbacks;
  124. #onRemovedCallbacks;
  125. #nodeObserver;
  126. #nodeWatchers;
  127. #textObserver;
  128. #onTextChangedCallbacks;
  129. #onAttrChangedCallbacks;
  130. visibilityAncestor;
  131. #visibilityObserver;
  132. #isVisible;
  133. #visibilityWatchers;
  134. constructor(name, elem) {
  135. this.name = name;
  136. this.element = null;
  137. this.#onCreatedCallbacks = [];
  138. this.#onRemovedCallbacks = [];
  139. this.#nodeObserver = null;
  140. this.#nodeWatchers = [];
  141. this.#textObserver = null;
  142. this.#onTextChangedCallbacks = [];
  143. this.#onAttrChangedCallbacks = [];
  144. this.visibilityAncestor = null;
  145. this.#visibilityObserver = null;
  146. this.#isVisible = null;
  147. this.#visibilityWatchers = [];
  148. if (elem != null) {
  149. this.#connect(elem);
  150. }
  151. }
  152. #assertElement() {
  153. if (this.element == null) {
  154. throw new Error(`Watcher not connected to an element`);
  155. }
  156. return this.element;
  157. }
  158. #assertVisibilityAncestor() {
  159. if (this.visibilityAncestor == null) {
  160. throw new Error(`Watcher is missing a visibilityAncestor`);
  161. }
  162. return this.visibilityAncestor;
  163. }
  164. #connect(element, visibilityAncestor) {
  165. if (this.element != null) {
  166. if (this.element !== element) {
  167. console.error(
  168. logPrefix,
  169. `Watcher already connected to`,
  170. this.element,
  171. `while trying to connect to`,
  172. element
  173. );
  174. }
  175. return;
  176. }
  177. this.element = element;
  178. this.visibilityAncestor = visibilityAncestor ?? null;
  179. for (const onCreatedCb of this.#onCreatedCallbacks) {
  180. const onRemovedCb = onCreatedCb(this.element);
  181. if (onRemovedCb) {
  182. this.#onRemovedCallbacks.push(onRemovedCb);
  183. }
  184. }
  185. for (const { selector, name, watcher: watcher2 } of this.#nodeWatchers) {
  186. for (const descElem of getDescendantsBy(this.element, selector, name)) {
  187. watcher2.#connect(descElem, this.element);
  188. }
  189. }
  190. for (const callback of this.#onTextChangedCallbacks) {
  191. callback(this.element, this.element.textContent);
  192. }
  193. for (const { name, callback } of this.#onAttrChangedCallbacks) {
  194. callback(this.element, this.element.getAttribute(name));
  195. }
  196. this.#registerNodeObserver();
  197. this.#registerTextObserver();
  198. this.#registerAttrObservers();
  199. this.#registerVisibilityObserver();
  200. }
  201. #disconnect() {
  202. if (this.element == null) {
  203. return;
  204. }
  205. for (const child of this.#nodeWatchers) {
  206. child.watcher.#disconnect();
  207. }
  208. for (const callback of this.#onTextChangedCallbacks) {
  209. callback(this.element, void 0);
  210. }
  211. for (const { callback } of this.#onAttrChangedCallbacks) {
  212. callback(this.element, void 0);
  213. }
  214. for (const child of this.#visibilityWatchers) {
  215. child.#disconnect();
  216. }
  217. this.#deregisterNodeObserver();
  218. this.#deregisterTextObserver();
  219. this.#deregisterAttrObservers();
  220. this.#deregisterVisibilityObserver();
  221. while (this.#onRemovedCallbacks.length > 0) {
  222. const onRemovedCb = this.#onRemovedCallbacks.shift();
  223. onRemovedCb();
  224. }
  225. this.element = null;
  226. }
  227. #registerNodeObserver() {
  228. if (this.#nodeObserver != null) {
  229. return;
  230. }
  231. if (this.#nodeWatchers.length === 0) {
  232. return;
  233. }
  234. const elem = this.#assertElement();
  235. this.#nodeObserver = new MutationObserver((mutations) => {
  236. for (const mut of mutations) {
  237. for (const node of mut.addedNodes) {
  238. for (const { selector, name, watcher: watcher2 } of this.#nodeWatchers) {
  239. for (const descElem of getSelfOrDescendantsBy(
  240. node,
  241. selector,
  242. name
  243. )) {
  244. watcher2.#connect(descElem, elem);
  245. }
  246. }
  247. }
  248. for (const node of mut.removedNodes) {
  249. for (const { selector, name, watcher: watcher2 } of this.#nodeWatchers) {
  250. for (const _descElem of getSelfOrDescendantsBy(
  251. node,
  252. selector,
  253. name
  254. )) {
  255. watcher2.#disconnect();
  256. }
  257. }
  258. }
  259. }
  260. });
  261. this.#nodeObserver.observe(elem, {
  262. subtree: true,
  263. childList: true
  264. });
  265. }
  266. #registerTextObserver() {
  267. if (this.#textObserver != null) {
  268. return;
  269. }
  270. if (this.#onTextChangedCallbacks.length === 0) {
  271. return;
  272. }
  273. const elem = this.#assertElement();
  274. this.#textObserver = new MutationObserver((_mutations) => {
  275. for (const callback of this.#onTextChangedCallbacks) {
  276. callback(elem, elem.textContent);
  277. }
  278. });
  279. this.#textObserver.observe(elem, {
  280. subtree: true,
  281. // This is needed when elements are replaced to update their text.
  282. childList: true,
  283. characterData: true
  284. });
  285. }
  286. #registerAttrObservers() {
  287. const elem = this.#assertElement();
  288. for (const handler of this.#onAttrChangedCallbacks) {
  289. if (handler.observer != null) {
  290. continue;
  291. }
  292. const { name, callback } = handler;
  293. handler.observer = new MutationObserver((_mutations) => {
  294. callback(elem, elem.getAttribute(name));
  295. });
  296. handler.observer.observe(elem, {
  297. attributes: true,
  298. attributeFilter: [name]
  299. });
  300. }
  301. }
  302. #registerVisibilityObserver() {
  303. if (this.#visibilityObserver != null) {
  304. return;
  305. }
  306. if (this.#visibilityWatchers.length === 0) {
  307. return;
  308. }
  309. this.#isVisible = false;
  310. const elem = this.#assertElement();
  311. const visibilityAncestor = this.#assertVisibilityAncestor();
  312. this.#visibilityObserver = new IntersectionObserver(
  313. (entries) => {
  314. const oldVisible = this.#isVisible;
  315. for (const entry of entries) {
  316. this.#isVisible = entry.isIntersecting;
  317. }
  318. if (this.#isVisible !== oldVisible) {
  319. if (this.#isVisible) {
  320. for (const watcher2 of this.#visibilityWatchers) {
  321. watcher2.#connect(elem, visibilityAncestor);
  322. }
  323. } else {
  324. for (const watcher2 of this.#visibilityWatchers) {
  325. watcher2.#disconnect();
  326. }
  327. }
  328. }
  329. },
  330. {
  331. root: visibilityAncestor
  332. }
  333. );
  334. this.#visibilityObserver.observe(elem);
  335. }
  336. #deregisterNodeObserver() {
  337. if (this.#nodeObserver == null) {
  338. return;
  339. }
  340. this.#nodeObserver.disconnect();
  341. this.#nodeObserver = null;
  342. }
  343. #deregisterTextObserver() {
  344. if (this.#textObserver == null) {
  345. return;
  346. }
  347. this.#textObserver.disconnect();
  348. this.#textObserver = null;
  349. }
  350. #deregisterAttrObservers() {
  351. for (const handler of this.#onAttrChangedCallbacks) {
  352. if (handler.observer == null) {
  353. continue;
  354. }
  355. handler.observer.disconnect();
  356. handler.observer = null;
  357. }
  358. }
  359. #deregisterVisibilityObserver() {
  360. if (this.#visibilityObserver == null) {
  361. return;
  362. }
  363. this.#visibilityObserver.disconnect();
  364. this.#visibilityObserver = null;
  365. this.#isVisible = null;
  366. }
  367. onCreated(onCreatedCb) {
  368. this.#onCreatedCallbacks.push(onCreatedCb);
  369. if (this.element != null) {
  370. const onRemovedCb = onCreatedCb(this.element);
  371. if (onRemovedCb) {
  372. this.#onRemovedCallbacks.push(onRemovedCb);
  373. }
  374. }
  375. return this;
  376. }
  377. descendant(selector, name) {
  378. const watcher2 = new Watcher(`${this.name} ${name}`);
  379. this.#nodeWatchers.push({ selector, name, watcher: watcher2 });
  380. if (this.element != null) {
  381. for (const descElem of getDescendantsBy(this.element, selector, name)) {
  382. watcher2.#connect(descElem, this.element);
  383. }
  384. this.#registerNodeObserver();
  385. }
  386. return watcher2;
  387. }
  388. id(idName) {
  389. return this.descendant("id", idName);
  390. }
  391. klass(className) {
  392. return this.descendant("class", className);
  393. }
  394. tag(tagName) {
  395. return this.descendant("tag", tagName);
  396. }
  397. visible() {
  398. const watcher2 = new Watcher(`${this.name} (visible)`);
  399. this.#visibilityWatchers.push(watcher2);
  400. if (this.element != null) {
  401. const visibilityAncestor = this.#assertVisibilityAncestor();
  402. if (this.#isVisible) {
  403. watcher2.#connect(this.element, visibilityAncestor);
  404. }
  405. this.#registerVisibilityObserver();
  406. }
  407. return watcher2;
  408. }
  409. /// `null` implies null textContent. `undefined` implies that the watcher is
  410. /// being disconnected.
  411. text(callback) {
  412. this.#onTextChangedCallbacks.push(callback);
  413. if (this.element != null) {
  414. callback(this.element, this.element.textContent);
  415. this.#registerTextObserver();
  416. }
  417. return this;
  418. }
  419. /// `null` implies no such attribute. `undefined` implies that the watcher is
  420. /// being disconnected.
  421. attr(name, callback) {
  422. this.#onAttrChangedCallbacks.push({ name, callback, observer: null });
  423. if (this.element != null) {
  424. callback(this.element, this.element.getAttribute(name));
  425. this.#registerAttrObservers();
  426. }
  427. return this;
  428. }
  429. }
  430. function getSelfOrDescendantsBy(node, selector, name) {
  431. if (!(node instanceof HTMLElement)) {
  432. return [];
  433. }
  434. if (selector === "id" || selector === "class" || selector === "tag") {
  435. if (selector === "id" && node.id === name || selector === "class" && node.classList.contains(name) || selector === "tag" && node.tagName.toLowerCase() === name.toLowerCase()) {
  436. return [node];
  437. } else {
  438. return getDescendantsBy(node, selector, name);
  439. }
  440. } else {
  441. const impossible = selector;
  442. throw new Error(`Impossible selector type: ${JSON.stringify(impossible)}`);
  443. }
  444. }
  445. function getDescendantsBy(node, selector, name) {
  446. if (!(node instanceof HTMLElement)) {
  447. return [];
  448. }
  449. let cssSelector = "";
  450. if (selector === "id") {
  451. cssSelector += "#";
  452. } else if (selector === "class") {
  453. cssSelector += ".";
  454. } else if (selector === "tag") ;
  455. else {
  456. const impossible = selector;
  457. throw new Error(`Impossible selector type: ${JSON.stringify(impossible)}`);
  458. }
  459. cssSelector += CSS.escape(name);
  460. return Array.from(node.querySelectorAll(cssSelector));
  461. }
  462. const videoSelector = "#movie_player video";
  463. function getVideoElement() {
  464. const videoElem = document.querySelector(videoSelector);
  465. if (!(videoElem instanceof HTMLVideoElement)) {
  466. console.error(
  467. logPrefix,
  468. "Expected",
  469. JSON.stringify(videoSelector),
  470. "to be a video element, got:",
  471. videoElem?.cloneNode(true)
  472. );
  473. return null;
  474. }
  475. return videoElem;
  476. }
  477. function callMoviePlayerMethod(name, onSuccess, args) {
  478. try {
  479. const movieElem = document.getElementById("movie_player");
  480. if (movieElem == null) {
  481. console.warn(logPrefix, "movie_player element not found");
  482. return;
  483. }
  484. const method = Object.getOwnPropertyDescriptor(
  485. movieElem,
  486. name
  487. )?.value;
  488. if (method == null) {
  489. console.warn(
  490. logPrefix,
  491. `movie_player element has no ${JSON.stringify(name)} property`
  492. );
  493. return;
  494. }
  495. if (!(typeof method === "function")) {
  496. console.warn(
  497. logPrefix,
  498. `movie_player element property ${JSON.stringify(name)} is not a function`
  499. );
  500. return;
  501. }
  502. const result = method.apply(movieElem, args);
  503. if (onSuccess != null) {
  504. onSuccess(result);
  505. }
  506. return result;
  507. } catch (e) {
  508. console.warn(
  509. logPrefix,
  510. `movie_player method ${JSON.stringify(name)} failed:`,
  511. e
  512. );
  513. return;
  514. }
  515. }
  516. function disableVisibilityChecks() {
  517. for (const eventName of ["visibilitychange", "blur", "focus"]) {
  518. document.addEventListener(
  519. eventName,
  520. (ev) => {
  521. ev.stopImmediatePropagation();
  522. },
  523. { capture: true }
  524. );
  525. }
  526. document.hasFocus = () => true;
  527. Object.defineProperties(document, {
  528. visibilityState: { value: "visible" },
  529. hidden: { value: false }
  530. });
  531. }
  532. function adIsPlaying(_elem) {
  533. console.info(logPrefix, "An ad is playing, muting and speeding up");
  534. const video = getVideoElement();
  535. if (video == null) {
  536. return;
  537. }
  538. const onRemovedCallbacks = [
  539. mute(video),
  540. speedup(video),
  541. cancelPlayback(video)
  542. ];
  543. return function onRemoved() {
  544. for (const callback of onRemovedCallbacks) {
  545. callback();
  546. }
  547. };
  548. }
  549. function mute(video) {
  550. console.debug(logPrefix, "Muting");
  551. video.muted = true;
  552. return unmute;
  553. }
  554. function unmute() {
  555. {
  556. return;
  557. }
  558. }
  559. function speedup(video) {
  560. for (let rate = 16; rate >= 2; rate /= 2) {
  561. try {
  562. video.playbackRate = rate;
  563. break;
  564. } catch (e) {
  565. console.debug(logPrefix, `Setting playback rate to`, rate, `failed:`, e);
  566. }
  567. }
  568. return function onRemoved() {
  569. const originalRate = callMoviePlayerMethod("getPlaybackRate");
  570. if (originalRate == null || typeof originalRate !== "number" || isNaN(originalRate)) {
  571. console.warn(
  572. logPrefix,
  573. `Restoring playback rate failed:`,
  574. `unable to query the current playback rate, got: ${JSON.stringify(originalRate)}.`,
  575. `Falling back to 1.`
  576. );
  577. restorePlaybackRate(video, 1);
  578. return;
  579. }
  580. restorePlaybackRate(video, originalRate);
  581. };
  582. }
  583. function restorePlaybackRate(video, originalRate) {
  584. try {
  585. video.playbackRate = originalRate;
  586. } catch (e) {
  587. console.debug(
  588. logPrefix,
  589. `Restoring playback rate to`,
  590. originalRate,
  591. `failed:`,
  592. e
  593. );
  594. }
  595. }
  596. function cancelPlayback(video) {
  597. let shouldResume = false;
  598. function doCancelPlayback() {
  599. console.info(logPrefix, "Attempting to cancel playback");
  600. callMoviePlayerMethod("cancelPlayback", () => {
  601. shouldResume = true;
  602. });
  603. }
  604. if (video.paused) {
  605. console.debug(
  606. logPrefix,
  607. "Ad paused, waiting for it to play before canceling playback"
  608. );
  609. video.addEventListener("play", doCancelPlayback);
  610. } else {
  611. doCancelPlayback();
  612. }
  613. return function onRemoved() {
  614. video.removeEventListener("play", doCancelPlayback);
  615. if (shouldResume) {
  616. resumePlaybackIfNotAtEnd();
  617. }
  618. };
  619. }
  620. function resumePlaybackIfNotAtEnd() {
  621. const currentTime = callMoviePlayerMethod("getCurrentTime");
  622. const duration = callMoviePlayerMethod("getDuration");
  623. const isAtLiveHead = callMoviePlayerMethod("isAtLiveHead");
  624. if (currentTime == null || duration == null || typeof currentTime !== "number" || typeof duration !== "number" || isNaN(currentTime) || isNaN(duration)) {
  625. console.warn(
  626. logPrefix,
  627. `movie_player methods getCurrentTime/getDuration failed, got time: ${JSON.stringify(currentTime)}, duration: ${JSON.stringify(duration)}`
  628. );
  629. return;
  630. }
  631. if (isAtLiveHead == null || typeof isAtLiveHead !== "boolean") {
  632. console.warn(
  633. logPrefix,
  634. `movie_player method isAtLiveHead failed, got: ${JSON.stringify(isAtLiveHead)}`
  635. );
  636. return;
  637. }
  638. const atEnd = duration - currentTime < 1;
  639. if (atEnd && !isAtLiveHead) {
  640. console.info(
  641. logPrefix,
  642. `Video is at the end (${currentTime}/${duration}), not attempting to resume playback`
  643. );
  644. return;
  645. }
  646. console.info(logPrefix, "Attempting to resume playback");
  647. callMoviePlayerMethod("playVideo");
  648. }
  649. function click(description) {
  650. return (elem) => {
  651. if (elem.getAttribute("aria-hidden")) {
  652. console.info(logPrefix, "Not clicking (aria-hidden):", description);
  653. } else {
  654. console.info(logPrefix, "Clicking:", description);
  655. elem.click();
  656. }
  657. };
  658. }
  659. disableVisibilityChecks();
  660. const watcher = new Watcher("body", document.body);
  661. const adPlayerOverlayClasses = [
  662. "ytp-ad-player-overlay",
  663. "ytp-ad-player-overlay-layout"
  664. // Seen since 2024-04-06.
  665. ];
  666. for (const adPlayerOverlayClass of adPlayerOverlayClasses) {
  667. watcher.klass(adPlayerOverlayClass).onCreated(adIsPlaying);
  668. }
  669. const adSkipButtonClasses = [
  670. "ytp-ad-skip-button",
  671. "ytp-ad-skip-button-modern",
  672. // Seen since 2023-11-10.
  673. "ytp-skip-ad-button"
  674. // Seen since 2024-04-06.
  675. ];
  676. for (const adSkipButtonClass of adSkipButtonClasses) {
  677. watcher.id("movie_player").klass(adSkipButtonClass).visible().attr("aria-hidden", (elem, value) => {
  678. if (value === null) {
  679. click(`skip (${adSkipButtonClass})`)(elem);
  680. }
  681. });
  682. }
  683. watcher.klass("ytp-ad-overlay-close-button").visible().onCreated(click("overlay close"));
  684. watcher.tag("ytmusic-you-there-renderer").tag("button").visible().onCreated(click("are-you-there"));
  685.  
  686. })();