- // ==UserScript==
- // @name Stream Skipper
- // @name:ja Stream Skipper
- // @name:en Stream Skipper
- // @name:zh-CN Stream Skipper
- // @name:ko Stream Skipper
- // @name:ru Stream Skipper
- // @name:de Stream Skipper
- // @description skip intro and ending
- // @description:ja イントロとエンディングをスキップする
- // @description:en skip intro and ending
- // @description:zh-CN 跳过介绍和结尾
- // @description:ko 인트로와 엔딩을 스킵합니다
- // @description:ru пропускает интро и окончание
- // @description:de Überspringt Intro und Ende
- // @version 2.3.7
- // @author Yos_sy
- // @match *://*.amazon.com/*
- // @match *://*.amazon.ca/*
- // @match *://*.amazon.com.mx/*
- // @match *://*.amazon.co.uk/*
- // @match *://*.amazon.de/*
- // @match *://*.amazon.fr/*
- // @match *://*.amazon.it/*
- // @match *://*.amazon.es/*
- // @match *://*.amazon.nl/*
- // @match *://*.amazon.se/*
- // @match *://*.amazon.pl/*
- // @match *://*.amazon.co.jp/*
- // @match *://*.amazon.com.au/*
- // @match *://*.amazon.in/*
- // @match *://*.amazon.cn/*
- // @match *://*.amazon.com.br/*
- // @match *://*.amazon.sa/*
- // @match *://*.amazon.ae/*
- // @match *://*.amazon.sg/*
- // @match *://*.amazon.com.tr/*
- // @match *://*.www.netflix.com/*
- // @namespace http://tampermonkey.net/
- // @icon 
- // @license MIT
- // @grant GM_setValue
- // @grant GM_getValue
- // ==/UserScript==
-
- (function () {
- "use strict";
-
- class IntroEndingSkipper {
- constructor() {
- this.skipIntroEnabled = this.loadState("skipIntroEnabled", true);
- this.skipEndingEnabled = this.loadState("skipEndingEnabled", true);
- this.skipEnabled = this.loadState("skipEnabled", true);
- this.initToggleButton();
- this.setupShortcut();
- this.updateToggleButton();
- this.initHUD();
- this.setupFullscreenHandler();
-
- this.buttonSelectors = {
- primeVideo: {
- intro: {
- type: "single",
- delay: 1500,
- selector:
- "button.fqye4e3.f1ly7q5u.fk9c3ap.fz9ydgy.f1xrlb00.f1hy0e6n.fgbpje3.f1uteees.f1h2a8xb.atvwebplayersdk-skipelement-button.fjgzbz9.fiqc9rt.fg426ew.f1ekwadg",
- },
- ending: {
- type: "multi",
- delay: 1500,
- selector:
- "div.atvwebplayersdk-nextupcard-button.fixbm5z.f1nog967.fobx3y5",
- offSelector:
- "div.fxviu8c > button.atvwebplayersdk-nextupcardhide-button",
- },
- nextEpisode: {
- selector: "button.atvwebplayersdk-nexttitle-button",
- },
- },
- netflix: {
- intro: {
- type: "single",
- delay: 0,
- selector: "button.watch-video--skip-content-button",
- },
- ending: {
- type: "multi",
- delay: 0,
- selector: "button[data-uia='next-episode-seamless-button']",
- offSelector: "button[data-uia='watch-credits-seamless-button']",
- },
- nextEpisode: {
- selector: "button[data-uia='control-next']",
- },
- },
- };
- }
-
- // ローカルストレージから状態を読み込み
- loadState(key, defaultValue) {
- const storedState = GM_getValue(key, null);
- return storedState === null ? defaultValue : storedState;
- }
-
- // ローカルストレージに保存
- saveState(key, value) {
- GM_setValue(key, value);
- }
-
- // イントロをスキップ
- skipIntro() {
- for (const service in this.buttonSelectors) {
- const { selector, delay } = this.buttonSelectors[service].intro;
-
- if (this.skipEnabled && this.skipIntroEnabled) {
- const introButton = document.querySelector(selector);
- if (introButton) {
- setTimeout(() => {
- introButton.click();
- console.log(`Intro skipped for ${service}`);
- }, delay);
- break;
- }
- }
- }
- }
-
- // エンディングをスキップ
- skipEnding() {
- for (const service in this.buttonSelectors) {
- const { selector, offSelector, delay } =
- this.buttonSelectors[service].ending;
-
- if (this.skipEnabled && this.skipEndingEnabled) {
- // ON の場合 selector をクリック
- const endingButton = document.querySelector(selector);
- if (endingButton) {
- setTimeout(() => {
- endingButton.click();
- console.log(`Ending skipped for ${service}`);
- }, delay);
- }
- } else {
- // OFF の場合 offSelector をクリック
- const offButton = document.querySelector(offSelector);
- if (offButton) {
- setTimeout(() => {
- offButton.click();
- console.log(`Watch credits for ${service}`);
- }, delay);
- }
- }
- }
- }
-
- // 次のエピソードボタンをクリック
- clickNextEpisode() {
- for (const service in this.buttonSelectors) {
- if (this.buttonSelectors[service].nextEpisode) {
- const { selector } = this.buttonSelectors[service].nextEpisode;
- const nextButton = document.querySelector(selector);
- if (nextButton) {
- nextButton.click();
- console.log(`Next episode clicked for ${service}`);
- break;
- }
- }
- }
- }
-
- // トグルボタンを初期化
- initToggleButton() {
- this.toggleButton = document.createElement("div");
- this.toggleButton.textContent = this.skipEnabled
- ? "Skip: ON"
- : "Skip: OFF";
- this.toggleButton.style.cssText = `
- position: fixed;
- bottom: 24px;
- right: 24px;
- z-index: 2147483647;
- color: #fff;
- background-color: ${this.skipEnabled ? "rgba(0, 128, 0, 0.7)" : "rgba(255, 0, 0, 0.7)"};
- border: 2px solid #fff;
- padding: 10px 20px;
- border-radius: 25px;
- font: bold 16px/1.6 Arial, sans-serif;
- transition: opacity 0.3s ease-in-out;
- box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2);
- display: none;
- `;
- document.body.appendChild(this.toggleButton);
- }
-
- // トグルボタンの表示を更新
- updateToggleButton() {
- this.toggleButton.textContent = this.skipEnabled
- ? "Skip: ON"
- : "Skip: OFF";
- this.toggleButton.style.backgroundColor = this.skipEnabled
- ? "rgba(0, 128, 0, 0.7)"
- : "rgba(255, 0, 0, 0.7)";
- }
-
- // スキップ機能の全体的な切り替え
- toggleSkipping() {
- this.skipEnabled = !this.skipEnabled;
- this.saveState("skipEnabled", this.skipEnabled);
- this.updateToggleButton();
-
- this.toggleButton.style.display = "block";
- this.toggleButton.style.opacity = "1";
-
- setTimeout(() => {
- this.toggleButton.style.opacity = "0";
- setTimeout(() => {
- this.toggleButton.style.display = "none";
- }, 100);
- }, 1000);
-
- console.log(`Skipping is ${this.skipEnabled ? "enabled" : "disabled"}`);
- }
-
- // イントロスキップの切り替え
- toggleSkipIntro() {
- if (!this.skipEnabled) return;
- this.skipIntroEnabled = !this.skipIntroEnabled;
- this.saveState("skipIntroEnabled", this.skipIntroEnabled);
- this.updateHUD();
-
- console.log(
- `Intro Skipping is ${this.skipIntroEnabled ? "enabled" : "disabled"}`
- );
- }
-
- // エンディングスキップの切り替え
- toggleSkipEnding() {
- if (!this.skipEnabled) return;
- this.skipEndingEnabled = !this.skipEndingEnabled;
- this.saveState("skipEndingEnabled", this.skipEndingEnabled);
- this.updateHUD();
-
- console.log(
- `Ending Skipping is ${this.skipEndingEnabled ? "enabled" : "disabled"}`
- );
- }
-
- // キーボードショートカットの設定
- setupShortcut() {
- document.addEventListener("keydown", (event) => {
- if (event.altKey && event.key === "z") this.toggleSkipping();
- else if (event.altKey && event.key === "x") this.toggleSkipIntro();
- else if (event.altKey && event.key === "c") this.toggleSkipEnding();
- else if (event.key === "n") this.clickNextEpisode();
- });
- }
-
- // HUDを初期化
- initHUD() {
- this.hudElement = document.createElement("div");
- this.hudElement.style.cssText = `
- position: fixed;
- top: 24px;
- left: 24px;
- z-index: 2147483647;
- color: #fff;
- background: #000000CC;
- padding: 16px 24px;
- border-radius: 16px;
- font: 16px/1.6 Arial, sans-serif;
- transition: opacity 0.3s ease-in-out;
- box-shadow: rgba(0, 0, 0, 0.3) 0px 4px 8px;
- text-align: center;
- display: none;
- `;
- document.body.appendChild(this.hudElement);
- }
-
- // HUDを更新
- updateHUD() {
- this.hudElement.innerHTML = `
- <strong>Status</strong><br>
- Intro: ${this.skipIntroEnabled ? "ON" : "OFF"}<br>
- Ending: ${this.skipEndingEnabled ? "ON" : "OFF"}
- `;
- this.hudElement.style.display = "block";
- this.hudElement.style.opacity = "1";
- setTimeout(() => {
- this.hudElement.style.opacity = "0";
- setTimeout(() => {
- this.hudElement.style.display = "none";
- }, 100);
- }, 1000);
- }
-
- // DOMの変更を監視し、スキップ機能を適用
- observe() {
- const observer = new MutationObserver(() => {
- if (this.skipEnabled) {
- this.skipIntro();
- this.skipEnding();
- }
- });
- observer.observe(document.body, { childList: true, subtree: true });
- }
-
- setupFullscreenHandler() {
- document.addEventListener("fullscreenchange", () => {
- if (document.fullscreenElement) {
- this.moveElementsToFullscreen();
- } else {
- this.restoreElementsPosition();
- }
- });
- }
-
- moveElementsToFullscreen() {
- const fullscreenElement = document.fullscreenElement;
- if (fullscreenElement) {
- fullscreenElement.appendChild(this.toggleButton);
- fullscreenElement.appendChild(this.hudElement);
- }
- }
-
- restoreElementsPosition() {
- document.body.appendChild(this.toggleButton);
- document.body.appendChild(this.hudElement);
- }
- }
-
- const skipper = new IntroEndingSkipper();
- skipper.observe();
- })();