Greasy Fork

Stream Skipper

跳过介绍和结尾

// ==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.6
// @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: "single",
            delay: 1500,
            selector:
              "div.atvwebplayersdk-nextupcard-button.fixbm5z.f1nog967.fobx3y5",
          },
          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();
})();