Stream Skipper

skip intro and ending

  1. // ==UserScript==
  2. // @name Stream Skipper
  3. // @name:ja Stream Skipper
  4. // @name:en Stream Skipper
  5. // @name:zh-CN Stream Skipper
  6. // @name:ko Stream Skipper
  7. // @name:ru Stream Skipper
  8. // @name:de Stream Skipper
  9. // @description skip intro and ending
  10. // @description:ja イントロとエンディングをスキップする
  11. // @description:en skip intro and ending
  12. // @description:zh-CN 跳过介绍和结尾
  13. // @description:ko 인트로와 엔딩을 스킵합니다
  14. // @description:ru пропускает интро и окончание
  15. // @description:de Überspringt Intro und Ende
  16. // @version 2.3.7
  17. // @author Yos_sy
  18. // @match *://*.amazon.com/*
  19. // @match *://*.amazon.ca/*
  20. // @match *://*.amazon.com.mx/*
  21. // @match *://*.amazon.co.uk/*
  22. // @match *://*.amazon.de/*
  23. // @match *://*.amazon.fr/*
  24. // @match *://*.amazon.it/*
  25. // @match *://*.amazon.es/*
  26. // @match *://*.amazon.nl/*
  27. // @match *://*.amazon.se/*
  28. // @match *://*.amazon.pl/*
  29. // @match *://*.amazon.co.jp/*
  30. // @match *://*.amazon.com.au/*
  31. // @match *://*.amazon.in/*
  32. // @match *://*.amazon.cn/*
  33. // @match *://*.amazon.com.br/*
  34. // @match *://*.amazon.sa/*
  35. // @match *://*.amazon.ae/*
  36. // @match *://*.amazon.sg/*
  37. // @match *://*.amazon.com.tr/*
  38. // @match *://*.www.netflix.com/*
  39. // @namespace http://tampermonkey.net/
  40. // @icon 
  41. // @license MIT
  42. // @grant GM_setValue
  43. // @grant GM_getValue
  44. // ==/UserScript==
  45.  
  46. (function () {
  47. "use strict";
  48.  
  49. class IntroEndingSkipper {
  50. constructor() {
  51. this.skipIntroEnabled = this.loadState("skipIntroEnabled", true);
  52. this.skipEndingEnabled = this.loadState("skipEndingEnabled", true);
  53. this.skipEnabled = this.loadState("skipEnabled", true);
  54. this.initToggleButton();
  55. this.setupShortcut();
  56. this.updateToggleButton();
  57. this.initHUD();
  58. this.setupFullscreenHandler();
  59.  
  60. this.buttonSelectors = {
  61. primeVideo: {
  62. intro: {
  63. type: "single",
  64. delay: 1500,
  65. selector:
  66. "button.fqye4e3.f1ly7q5u.fk9c3ap.fz9ydgy.f1xrlb00.f1hy0e6n.fgbpje3.f1uteees.f1h2a8xb.atvwebplayersdk-skipelement-button.fjgzbz9.fiqc9rt.fg426ew.f1ekwadg",
  67. },
  68. ending: {
  69. type: "multi",
  70. delay: 1500,
  71. selector:
  72. "div.atvwebplayersdk-nextupcard-button.fixbm5z.f1nog967.fobx3y5",
  73. offSelector:
  74. "div.fxviu8c > button.atvwebplayersdk-nextupcardhide-button",
  75. },
  76. nextEpisode: {
  77. selector: "button.atvwebplayersdk-nexttitle-button",
  78. },
  79. },
  80. netflix: {
  81. intro: {
  82. type: "single",
  83. delay: 0,
  84. selector: "button.watch-video--skip-content-button",
  85. },
  86. ending: {
  87. type: "multi",
  88. delay: 0,
  89. selector: "button[data-uia='next-episode-seamless-button']",
  90. offSelector: "button[data-uia='watch-credits-seamless-button']",
  91. },
  92. nextEpisode: {
  93. selector: "button[data-uia='control-next']",
  94. },
  95. },
  96. };
  97. }
  98.  
  99. // ローカルストレージから状態を読み込み
  100. loadState(key, defaultValue) {
  101. const storedState = GM_getValue(key, null);
  102. return storedState === null ? defaultValue : storedState;
  103. }
  104.  
  105. // ローカルストレージに保存
  106. saveState(key, value) {
  107. GM_setValue(key, value);
  108. }
  109.  
  110. // イントロをスキップ
  111. skipIntro() {
  112. for (const service in this.buttonSelectors) {
  113. const { selector, delay } = this.buttonSelectors[service].intro;
  114.  
  115. if (this.skipEnabled && this.skipIntroEnabled) {
  116. const introButton = document.querySelector(selector);
  117. if (introButton) {
  118. setTimeout(() => {
  119. introButton.click();
  120. console.log(`Intro skipped for ${service}`);
  121. }, delay);
  122. break;
  123. }
  124. }
  125. }
  126. }
  127.  
  128. // エンディングをスキップ
  129. skipEnding() {
  130. for (const service in this.buttonSelectors) {
  131. const { selector, offSelector, delay } =
  132. this.buttonSelectors[service].ending;
  133.  
  134. if (this.skipEnabled && this.skipEndingEnabled) {
  135. // ON の場合 selector をクリック
  136. const endingButton = document.querySelector(selector);
  137. if (endingButton) {
  138. setTimeout(() => {
  139. endingButton.click();
  140. console.log(`Ending skipped for ${service}`);
  141. }, delay);
  142. }
  143. } else {
  144. // OFF の場合 offSelector をクリック
  145. const offButton = document.querySelector(offSelector);
  146. if (offButton) {
  147. setTimeout(() => {
  148. offButton.click();
  149. console.log(`Watch credits for ${service}`);
  150. }, delay);
  151. }
  152. }
  153. }
  154. }
  155.  
  156. // 次のエピソードボタンをクリック
  157. clickNextEpisode() {
  158. for (const service in this.buttonSelectors) {
  159. if (this.buttonSelectors[service].nextEpisode) {
  160. const { selector } = this.buttonSelectors[service].nextEpisode;
  161. const nextButton = document.querySelector(selector);
  162. if (nextButton) {
  163. nextButton.click();
  164. console.log(`Next episode clicked for ${service}`);
  165. break;
  166. }
  167. }
  168. }
  169. }
  170.  
  171. // トグルボタンを初期化
  172. initToggleButton() {
  173. this.toggleButton = document.createElement("div");
  174. this.toggleButton.textContent = this.skipEnabled
  175. ? "Skip: ON"
  176. : "Skip: OFF";
  177. this.toggleButton.style.cssText = `
  178. position: fixed;
  179. bottom: 24px;
  180. right: 24px;
  181. z-index: 2147483647;
  182. color: #fff;
  183. background-color: ${this.skipEnabled ? "rgba(0, 128, 0, 0.7)" : "rgba(255, 0, 0, 0.7)"};
  184. border: 2px solid #fff;
  185. padding: 10px 20px;
  186. border-radius: 25px;
  187. font: bold 16px/1.6 Arial, sans-serif;
  188. transition: opacity 0.3s ease-in-out;
  189. box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2);
  190. display: none;
  191. `;
  192. document.body.appendChild(this.toggleButton);
  193. }
  194.  
  195. // トグルボタンの表示を更新
  196. updateToggleButton() {
  197. this.toggleButton.textContent = this.skipEnabled
  198. ? "Skip: ON"
  199. : "Skip: OFF";
  200. this.toggleButton.style.backgroundColor = this.skipEnabled
  201. ? "rgba(0, 128, 0, 0.7)"
  202. : "rgba(255, 0, 0, 0.7)";
  203. }
  204.  
  205. // スキップ機能の全体的な切り替え
  206. toggleSkipping() {
  207. this.skipEnabled = !this.skipEnabled;
  208. this.saveState("skipEnabled", this.skipEnabled);
  209. this.updateToggleButton();
  210.  
  211. this.toggleButton.style.display = "block";
  212. this.toggleButton.style.opacity = "1";
  213.  
  214. setTimeout(() => {
  215. this.toggleButton.style.opacity = "0";
  216. setTimeout(() => {
  217. this.toggleButton.style.display = "none";
  218. }, 100);
  219. }, 1000);
  220.  
  221. console.log(`Skipping is ${this.skipEnabled ? "enabled" : "disabled"}`);
  222. }
  223.  
  224. // イントロスキップの切り替え
  225. toggleSkipIntro() {
  226. if (!this.skipEnabled) return;
  227. this.skipIntroEnabled = !this.skipIntroEnabled;
  228. this.saveState("skipIntroEnabled", this.skipIntroEnabled);
  229. this.updateHUD();
  230.  
  231. console.log(
  232. `Intro Skipping is ${this.skipIntroEnabled ? "enabled" : "disabled"}`
  233. );
  234. }
  235.  
  236. // エンディングスキップの切り替え
  237. toggleSkipEnding() {
  238. if (!this.skipEnabled) return;
  239. this.skipEndingEnabled = !this.skipEndingEnabled;
  240. this.saveState("skipEndingEnabled", this.skipEndingEnabled);
  241. this.updateHUD();
  242.  
  243. console.log(
  244. `Ending Skipping is ${this.skipEndingEnabled ? "enabled" : "disabled"}`
  245. );
  246. }
  247.  
  248. // キーボードショートカットの設定
  249. setupShortcut() {
  250. document.addEventListener("keydown", (event) => {
  251. if (event.altKey && event.key === "z") this.toggleSkipping();
  252. else if (event.altKey && event.key === "x") this.toggleSkipIntro();
  253. else if (event.altKey && event.key === "c") this.toggleSkipEnding();
  254. else if (event.key === "n") this.clickNextEpisode();
  255. });
  256. }
  257.  
  258. // HUDを初期化
  259. initHUD() {
  260. this.hudElement = document.createElement("div");
  261. this.hudElement.style.cssText = `
  262. position: fixed;
  263. top: 24px;
  264. left: 24px;
  265. z-index: 2147483647;
  266. color: #fff;
  267. background: #000000CC;
  268. padding: 16px 24px;
  269. border-radius: 16px;
  270. font: 16px/1.6 Arial, sans-serif;
  271. transition: opacity 0.3s ease-in-out;
  272. box-shadow: rgba(0, 0, 0, 0.3) 0px 4px 8px;
  273. text-align: center;
  274. display: none;
  275. `;
  276. document.body.appendChild(this.hudElement);
  277. }
  278.  
  279. // HUDを更新
  280. updateHUD() {
  281. this.hudElement.innerHTML = `
  282. <strong>Status</strong><br>
  283. Intro: ${this.skipIntroEnabled ? "ON" : "OFF"}<br>
  284. Ending: ${this.skipEndingEnabled ? "ON" : "OFF"}
  285. `;
  286. this.hudElement.style.display = "block";
  287. this.hudElement.style.opacity = "1";
  288. setTimeout(() => {
  289. this.hudElement.style.opacity = "0";
  290. setTimeout(() => {
  291. this.hudElement.style.display = "none";
  292. }, 100);
  293. }, 1000);
  294. }
  295.  
  296. // DOMの変更を監視し、スキップ機能を適用
  297. observe() {
  298. const observer = new MutationObserver(() => {
  299. if (this.skipEnabled) {
  300. this.skipIntro();
  301. this.skipEnding();
  302. }
  303. });
  304. observer.observe(document.body, { childList: true, subtree: true });
  305. }
  306.  
  307. setupFullscreenHandler() {
  308. document.addEventListener("fullscreenchange", () => {
  309. if (document.fullscreenElement) {
  310. this.moveElementsToFullscreen();
  311. } else {
  312. this.restoreElementsPosition();
  313. }
  314. });
  315. }
  316.  
  317. moveElementsToFullscreen() {
  318. const fullscreenElement = document.fullscreenElement;
  319. if (fullscreenElement) {
  320. fullscreenElement.appendChild(this.toggleButton);
  321. fullscreenElement.appendChild(this.hudElement);
  322. }
  323. }
  324.  
  325. restoreElementsPosition() {
  326. document.body.appendChild(this.toggleButton);
  327. document.body.appendChild(this.hudElement);
  328. }
  329. }
  330.  
  331. const skipper = new IntroEndingSkipper();
  332. skipper.observe();
  333. })();

QingJ © 2025

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