网飞马拉松赛(可暫停)

一个可配置的脚本,可自动跳过 Netflix、Amazon Prime Video 和 Disney+ 上的重述、介绍、字幕和广告,并单击“下一集”提示。 可自定义的热键来暂停/恢复自动跳过功能。 Alt + N 用于设置。

当前为 2021-09-30 提交的版本,查看 最新版本

// ==UserScript==
// @name               Netflix Marathon (Pausable)
// @name:en            Netflix Marathon (Pausable)
// @name:zh-CN         网飞马拉松赛(可暫停)
// @name:zh-TW         网飞马拉松赛(可暫停)
// @name:ja            Netflix Marathon(一時停止できます)
// @name:ko            Netflix 마라톤(일시 중지 가능)
// @name:ar            ماراثون Netflix (يمكن إيقافه مؤقتًا)
// @name:de            Netflix-Marathon (pausierbar)
// @name:ru            Netflix Marathon (пауза)
// @name:hi            नेटफ्लिक्स मैराथन (रोकने योग्य)
// @namespace          https://github.com/aminomancer
// @version            5.2.0
// @description        A configurable script that automatically skips recaps, intros, credits, and ads, and clicks "next episode" prompts on Netflix, Amazon Prime Video, and Disney+. Customizable hotkey to pause/resume the auto-skipping functionality. Alt + N for settings.
// @description:en     A configurable script that automatically skips recaps, intros, credits, and ads, and clicks "next episode" prompts on Netflix, Amazon Prime Video, and Disney+. Customizable hotkey to pause/resume the auto-skipping functionality. Alt + N for settings.
// @description:zh-CN  一个可配置的脚本,可自动跳过 Netflix、Amazon Prime Video 和 Disney+ 上的重述、介绍、字幕和广告,并单击“下一集”提示。 可自定义的热键来暂停/恢复自动跳过功能。 Alt + N 用于设置。
// @description:zh-TW  一个可配置的脚本,可自动跳过 Netflix、Amazon Prime Video 和 Disney+ 上的重述、介绍、字幕和广告,并单击“下一集”提示。 可自定义的热键来暂停/恢复自动跳过功能。 Alt + N 用于设置。
// @description:ja     要約、イントロ、クレジット、広告を自動的にスキップし、Netflix、Amazon Prime Video、Disney +の「次のエピソード」のプロンプトをクリックする構成可能なスクリプト。 自動スキップ機能を一時停止/再開するためのカスタマイズ可能なホットキー。 Alt + Nで設定します。
// @description:ko     요약, 소개, 크레딧 및 광고를 자동으로 건너뛰고 Netflix, Amazon Prime Video 및 Disney+에서 "다음 에피소드" 프롬프트를 클릭하는 구성 가능한 스크립트입니다. 자동 건너뛰기 기능을 일시 중지/재개하는 사용자 지정 가능한 단축키입니다. Alt + N은 설정입니다.
// @description:ar     برنامج نصي قابل للتكوين يتخطى الملخصات والمقدمات والاعتمادات والإعلانات تلقائيًا وينقر على "الحلقة التالية" على Netflix و Amazon Prime Video و Disney +. مفتاح التشغيل السريع القابل للتخصيص لإيقاف / استئناف وظيفة التخطي التلقائي. Alt + N للإعدادات.
// @description:de     Ein konfigurierbares Skript, das automatisch Zusammenfassungen, Vorspänne, Abspänne und Werbung überspringt und bei Netflix, Amazon Prime Video und Disney+ auf die Aufforderung "nächste Episode" klickt. Anpassbarer Hotkey zum Anhalten/Fortsetzen der Auto-Skipping-Funktion. Alt + N für Einstellungen.
// @description:ru     Настраиваемый сценарий, который автоматически пропускает резюме, вступление, титры и рекламу, а также нажимает подсказки «следующий выпуск» на Netflix, Amazon Prime Video и Disney +. Настраиваемая горячая клавиша для приостановки / возобновления функции автоматического пропуска. Alt + N для настроек.
// @description:hi     एक विन्यास योग्य स्क्रिप्ट जो स्वचालित रूप से रिकैप, इंट्रो, क्रेडिट और विज्ञापनों को छोड़ देती है, और नेटफ्लिक्स, अमेज़ॅन प्राइम वीडियो और डिज़नी + पर "अगला एपिसोड" पर क्लिक करती है। ऑटो-स्किपिंग कार्यक्षमता को रोकने/फिर से शुरू करने के लिए अनुकूलन योग्य हॉटकी। ऑल्ट + एन सेटिंग्स के लिए।
// @author             aminomancer
// @homepageURL        https://github.com/aminomancer/Netflix-Marathon-Pausable
// @supportURL         https://github.com/aminomancer/Netflix-Marathon-Pausable
// @icon               https://cdn.jsdelivr.net/gh/aminomancer/Netflix-Marathon-Pausable@latest/icon-small.svg
// @match              http*://*.amazon.ae/*
// @match              http*://*.amazon.ca/*
// @match              http*://*.amazon.cn/*
// @match              http*://*.amazon.co.jp/*
// @match              http*://*.amazon.co.uk/*
// @match              http*://*.amazon.com/*
// @match              http*://*.amazon.com.au/*
// @match              http*://*.amazon.com.br/*
// @match              http*://*.amazon.com.mx/*
// @match              http*://*.amazon.de/*
// @match              http*://*.amazon.eg/*
// @match              http*://*.amazon.es/*
// @match              http*://*.amazon.fr/*
// @match              http*://*.amazon.in/*
// @match              http*://*.amazon.it/*
// @match              http*://*.amazon.nl/*
// @match              http*://*.amazon.pl/*
// @match              http*://*.amazon.sa/*
// @match              http*://*.amazon.se/*
// @match              http*://*.amazon.sg/*
// @match              http*://*.amazon.tr/*
// @match              http*://*.disneyplus.com/*
// @match              http*://*.netflix.com/*
// @require            https://gf.qytechs.cn/scripts/420683-gm-config-sizzle/code/GM_config_sizzle.js?version=894369
// @grant              GM_registerMenuCommand
// @grant              GM_unregisterMenuCommand
// @grant              GM_setValue
// @grant              GM_getValue
// @grant              GM_deleteValue
// @grant              GM_listValues
// @grant              GM_openInTab
// @grant              GM.setValue
// @grant              GM.getValue
// @grant              GM.deleteValue
// @grant              GM.listValues
// @grant              GM.openInTab
// ==/UserScript==

/* global GM,
GM_registerMenuCommand,
GM_unregisterMenuCommand,
GM_getValue:writable,
GM_setValue:writable,
GM_deleteValue:writable,
GM_listValues:writable,
GM_openInTab:writable,
WebFontConfig:writable,
GM_config,
WebFont */
const options = {}; // where settings are stored during runtime
const win = window;
const doc = document;
// check whether the GM object exists so we can use the right GM API functions
const GMObj =
    "GM" in window && typeof window.GM === "object" && typeof window.GM.getValue === "function";
// check if the script handler is GM4, since if it is, we can't add a menu command
const GM4 = GMObj && GM.info.scriptHandler === "Greasemonkey" && GM.info.version.split(".")[0] >= 4;
let marathon;
/**
 * pause execution for ms milliseconds
 * @param {int} ms (milliseconds)
 */
const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
/**
 * @param {string} u (a string to test the URL against)
 */
const test = (u) => win.location.href.includes(u);
const getHost = () => {
    const urlParts = win.location.hostname.split(".");
    const host = urlParts
        .filter((part) => {
            switch (part) {
                case "amazon":
                case "primevideo":
                case "disneyplus":
                case "netflix":
                    return true;
                default:
                    return false;
            }
        })
        .join();
    switch (host) {
        case "amazon":
        case "primevideo":
            return "amazon";
        default:
            return host;
    }
};
const site = getHost();
const l10n = {
    // some basic localization for the settings menu. just the parts necessary to get to the readme, which has chinese, japanese, and arabic translations
    get lang() {
        return (
            this._lang || (this._lang = navigator.language.split("-")[0]) // memoize the language since it's unlikely to change during runtime
        );
    },
    get text() {
        // returns the label for the support button in settings
        if (this._text) return this._text;
        switch (this.lang) {
            case "zh":
                this._text = "信息"; // chinese
                break;
            case "ja":
                this._text = "助けて"; // japanese
                break;
            case "ko":
                this._text = "기술 지원"; // korean
                break;
            case "ar":
                this._text = "تعليمات"; // arabic
                break;
            case "de":
                this._text = "Hilfe"; // german
                break;
            case "ru":
                this._text = "помощь"; // russian
                break;
            case "hi":
                this._text = "तकनीकी समर्थन"; // hindi
                break;
            default:
                this._text = "Support"; // english etc.
        }
        return this._text;
    },
    get title() {
        // returns the tooltip for the support button
        if (this._title) return this._title;
        switch (this.lang) {
            case "zh":
                this._title = "设置的信息和翻译";
                break;
            case "ja":
                this._title = "設定の情報と翻訳";
                break;
            case "ko":
                this._title = "설정에 대한 정보 및 번역";
                break;
            case "ar":
                this._title = "معلومات وترجمات للإعدادات";
                break;
            case "de":
                this._title = "Infos und Übersetzungen zu den Einstellungen";
                break;
            case "ru":
                this._title = "Информация и переводы для настроек";
                break;
            case "hi":
                this._title = "सेटिंग्स के लिए जानकारी और अनुवाद";
                break;
            default:
                this._title = "Info and translations for the settings";
        }
        return this._title;
    },
};
const methods = {
    // contains the site-specific callbacks and various utility functions to shorten and optimize the code
    count: 0,
    results: null,
    nDrain: "[data-uia='next-episode-seamless-button-draining']",
    nReady: "[data-uia='next-episode-seamless-button']",
    /**
     * getElementsByTagName
     * @param {string} s (tag name)
     */
    byTag: (s, p = doc) => p.getElementsByTagName(s),
    /**
     * getElementById
     * @param {string} s (element id)
     */
    byID: (s) => doc.getElementById(s),
    /**
     * querySelector
     * @param {string} s (CSS selector e.g. ".class")
     */
    qry: (s, p = doc) => p.querySelector(s),
    /**
     * querySelectorAll
     * @param {string} s (CSS selector)
     */
    qryAll: (s, p = doc) => p.querySelectorAll(s),
    /**
     * document.evaluate
     * @param {string} s (node's text content)
     * @param {string} n (node's tag name. if not passed, then accept any tag)
     * @param {string} p (node's parent's tag name. this is like saying button>div. if not passed, then just use div, ignoring the node's parent)
     */
    byTxt(s, n = "*", p) {
        const exp = `//${p ? `${p}/child::` : ""}${n}[text()="${s}"]`; // use /child:: syntax if p is passed.
        return doc.evaluate(exp, doc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, this.results)
            .singleNodeValue;
    },
    /**
     * find react component instance given a DOM node
     * @param {object} d (DOM node)
     */
    react(d) {
        for (const [key, value] of Object.entries(d))
            if (key.startsWith("__reactInternalInstance$")) return value;
        return null;
    },
    /**
     * determine if an element is visible (namely the amazon player)
     * @param {string} s (element id)
     */
    isVis(s) {
        try {
            return !!this.byID(s).offsetParent;
        } catch (e) {
            return false;
        }
    },
    /**
     * clicks the passed element and sets the count to 5
     * @param {object} el (DOM element)
     */
    clk(el) {
        try {
            el.click();
            this.count = 5;
        } catch (e) {
            this.count = 2;
        }
    },
    /**
     * set a bunch of attributes on a node
     * @param {object} element (a DOM node)
     * @param {object} attrs (an object containing properties — keys are turned into attributes on the DOM node)
     */
    maybeSetAttributes(element, attrs) {
        for (const [name, value] of Object.entries(attrs))
            if (value === undefined) element.removeAttribute(name);
            else element.setAttribute(name, value);
    },
    /**
     * create a DOM node with given parameters
     * @param {object} aDoc (which doc to create the element in)
     * @param {string} tag (an HTML tag name, like "button" or "p")
     * @param {object} props (an object containing attribute name/value pairs, e.g. class: ".bookmark-item")
     * @returns the created DOM node
     */
    create(aDoc, tag, props) {
        const el = aDoc.createElement(tag);
        this.maybeSetAttributes(el, props);
        return el;
    },
    // searches for elements that skip stuff. repeated every 300ms. change "rate" in the options if you want to make this more or less frequent.
    async amazon() {
        if (this.count === 0) {
            if (this.isVis("dv-web-player")) {
                let store; // memoize the element when we check for its existence so we don't have to evaluate the DOM twice.
                if ((store = this.qry(".atvwebplayersdk-nextupcard-button"))) {
                    // next episode
                    await sleep(400);
                    this.clk(store);
                } else if ((store = this.qry(".atvwebplayersdk-skipelement-button")))
                    // skip various things
                    this.clk(store);
                else if ((store = this.qry(".adSkipButton")))
                    // skip ad
                    this.clk(store);
                else if ((store = this.qry(".skipElement")))
                    //  skip intro
                    this.clk(store);
                else if ((store = this.byTxt("Skip", "div")))
                    // skip trailers
                    this.clk(store);
                else if ((store = this.byTxt("Skip Intro", "button", "div")))
                    // skip intro
                    this.clk(store);
                else if ((store = this.byTxt("Skip Recap", "button", "div")))
                    // skip recap
                    this.clk(store);
            }
        } else this.count -= 1;
        return this.count;
    },
    async netflix() {
        if (this.count === 0) {
            let store;
            if (
                this.qryAll(".skip-credits").length &&
                this.qryAll(".skip-credits-hidden").length === 0
            ) {
                await sleep(200);
                try {
                    this.qry(".skip-credits").firstElementChild.click();
                    this.count = 80;
                } catch (e) {
                    return (this.count -= 1);
                }
                await sleep(100);
                try {
                    this.qry(".button-nfplayerPlay").click();
                    this.count = 80;
                } catch (e) {
                    return (this.count -= 1);
                }
            } else if ((store = this.qry(this.nDrain)) || (store = this.qry(this.nReady))) {
                // next episode button
                this.react(store).memoizedProps.onClick();
                this.count = 5;
            } else if (options.promoted && (store = this.qry(".PromotedVideo-actions"))) {
                // promoted video autoplay
                await sleep(700);
                this.clk(store.firstElementChild);
            } else if ((store = this.qry(".watch-video--skip-content-button")))
                // skip intro, recap, etc. (new netflix UI)
                this.clk(store);
            else if ((store = this.qry(".watch-video--skip-preplay-button")))
                // not sure what this does but I found this while trying to reverse engineer the source code. please inform me if you know
                this.clk(store);
            else if ((store = this.qry(".postplay-still-container")))
                // autoplay (old netflix UI)
                this.clk(store);
            else if ((store = this.qry(".WatchNext-still-container")))
                // autoplay (old netflix UI)
                this.clk(store);
        } else this.count -= 1;
        return this.count;
    },
    async disneyplus() {
        if (this.count === 0) {
            if (test("disneyplus.com/video/")) {
                let store;
                if ((store = this.qry(".skip__button")))
                    // skip intro, skip recap, skip credits, etc.
                    this.clk(store);
                else if ((store = this.qry(`button[data-gv2elementkey="playNext"]`))) {
                    // next episode
                    const spans = this.qryAll("span", store);
                    if (options.promoted || (spans && spans.length > 1)) this.clk(store);
                    // if there are 2 spans inside the button, it means the countdown number is visible.
                    // countdown number means it's going to automatically proceed in 10 seconds even if we don't click it,
                    // which usually happens when watching a series and proceeding to the next episode. so all we do in this case is speed up the process.
                    // but if there is only 1 span in the button, it means there's no countdown. it won't do anything without user interaction.
                    // this is the case when watching a film, and disney+ recommends a new title for the user to watch. hence, we read the "promoted" option.
                    // I looked hard for a cleaner, more future proof way to do this, even prying apart the react components. no such luck.
                }
            }
        } else this.count -= 1;
        return this.count;
    },
};

// an interval constructor that you can pause and resume, and which opens a brief popup when you do so. yes i'm using a class that's only instantiated once. i like the way it looks. if you know of something better lmk~
class Controller {
    /**
     * pausable interval utility
     * @param {object} handler (object containing the site methods)
     * @param {int} int (how often to repeat the callback)
     */
    constructor(handler, int) {
        this.callback = handler[site].bind(handler); // e.g. methods.amazon.bind(methods)
        this.int = int; // can be changed in real-time and the next resume() call will use the new value
        this.popup = doc.createElement("div");
        this.text = doc.createTextNode("Marathon: Paused");
        this.remainder = 0; // how much time is remaining on the interval when we pause it
        this.fading = null; // 3 second timeout (by default), after which the popup fades
        this.pauseState = 0; //  0: idle, 1: running, 2: paused, 3: resumed
        this.toggle = this.toggler.bind(this);
        this.register("Pause Marathon", true); // initial creation of the menu command
        // if popup is enabled in options, style it
        if (options.pop) {
            this.setupPopup();
            this.updatePopup();
        }
        this.time = new Date();
        this.timer = win.setInterval(this.callback, this.int);
        this.pauseState = 1;
        if (options.hotkey || options.hotkey2) this.startCapturing();
        if (!options[site]) this.pause(); // if the site is disabled then stop the interval. we pause it instead of not starting it in the first place so that the user can re-enable the site and have the interval immediately start working without needing to refresh the page.
    }

    /**
     * check that the modifier keys pressed match those defined in user settings
     * @param {object} e (event)
     * @param {string} d (which key settings to evaluate, ctrlKey or ctrlKey1)
     */
    static modTest(e, d = "") {
        return (
            e.ctrlKey === options[`ctrlKey${d}`] &&
            e.altKey === options[`altKey${d}`] &&
            e.shiftKey === options[`shiftKey${d}`] &&
            e.metaKey === options[`metaKey${d}`]
        );
    }

    /**
     * Controller's event handler. only handles keydown currently but may expand in the future.
     * @param {object} e (event)
     */
    handleEvent(e) {
        switch (e.type) {
            case "keydown":
                this.onKeyDown(e);
                break;
            default:
        }
    }

    /**
     * implementation for hotkeys and "escape to close"
     * @param {object} e (KeyboardEvent)
     */
    onKeyDown(e) {
        if (e.repeat) return;
        const { code, code2, hotkey, hotkey2 } = options;
        switch (e.code) {
            case code:
                if (hotkey && Controller.modTest(e)) this.toggle();
                else return;
                break;
            case code2:
                if (hotkey2 && Controller.modTest(e, 2))
                    GM_config.isOpen ? GM_config.close() : GM_config.open();
                else return;
                break;
            case "Escape":
                if (this.onEscape(e)) break;
                else return;
            default:
                return;
        }
        e.stopImmediatePropagation();
        e.preventDefault();
        e.stopPropagation();
    }

    /**
     * on pressing the Escape key, close any open popups
     * @param {object} e (KeyboardEvent)
     * @returns true if the config menu was open and we closed it. this determines whether the event will propagate any further or be consumed by the menu.
     */
    onEscape(e) {
        let consumed = false;
        if (e.ctrlKey || e.shiftKey || e.altKey || e.metaKey) return consumed;
        // hide the settings menu
        if (GM_config.isOpen) {
            GM_config.close();
            consumed = true;
        }
        // hide the pause/resume popup
        if (options.pop) {
            const { style } = this.popup;
            win.clearTimeout(this.fading);
            style.transitionDuration = "0.5s";
            style.opacity = "0";
        }
        return consumed;
    }

    /**
     * pause the interval
     * @param {string} pop (string or null — identifies the caller so we can determine the popup message)
     */
    pause(pop) {
        if (this.pauseState !== 1) return;
        this.remainder = this.int - (new Date() - this.time);
        win.clearInterval(this.timer);
        this.pauseState = 2;
        this.register("Resume Marathon"); // update the menu command label
        this.openPopup(pop);
    }

    /**
     * resume the interval
     * @param {string} pop (same as pause())
     */
    async resume(pop) {
        if (this.pauseState !== 2) return;
        this.pauseState = 3;
        this.register("Pause Marathon");
        this.openPopup(pop);
        await sleep(this.remainder);
        this.run();
    }

    // when we pause, there's usually still time left on the interval. resume() calls this after waiting for the remaining duration. so this is what actually resumes the interval.
    run() {
        if (this.pauseState !== 3) return;
        this.callback();
        this.time = new Date();
        this.timer = win.setInterval(this.callback, this.int);
        this.pauseState = 1;
    }

    // toggle the interval on/off.
    toggler() {
        if (!options[site]) return; // disable the pause/resume toggle when the site is disabled
        switch (this.pauseState) {
            case 1:
                this.pause("Paused"); // passing "Paused" tells openPopup to use the "Marathon: Paused" message
                break;
            case 2:
                this.resume("Resumed"); // passing "Resumed" => "Marathon: Resumed" message
                break;
            default:
        }
    }

    /**
     * opens the popup and schedules it to close
     * @param {string} msg (what the popup should say)
     */
    openPopup(msg) {
        // if popup is disabled in options, or no message was sent, do nothing
        if (msg === undefined || !options.pop) return;
        const { style } = this.popup;
        this.popup.textContent = `Marathon: ${msg}`;
        style.transitionDuration = "0.2s";
        style.opacity = "1";
        win.clearTimeout(this.fading); // clear any existing fade timeout since we're about to set a new one
        // schedule the popup to fade into oblivion
        this.fading = win.setTimeout(() => {
            style.transitionDuration = "1s";
            style.opacity = "0";
        }, options.popDur);
    }

    // apply the basic popup style and place it in the body
    setupPopup() {
        doc.body.insertBefore(this.popup, doc.body.firstElementChild);
        this.popup.appendChild(this.text);
        this.popup.style.cssText = `
            position: fixed;
            top: 50%;
            right: 3%;
            transform: translateY(-50%);
            z-index: 2147483646;
            background-color: hsla(0, 0%, 6%, 0.8);
            background-image: url("https://cdn.jsdelivr.net/gh/aminomancer/Netflix-Marathon-Pausable@latest/texture/noise-512x512.png");
            background-repeat: repeat;
            background-size: auto;
            background-attachment: local;
            -webkit-backdrop-filter: blur(7px);
            backdrop-filter: blur(7px);
            color: hsla(0, 0%, 97%, 0.95);
            padding: 17px 19px;
            line-height: 1em;
            border-radius: 5px;
            pointer-events: none;
            letter-spacing: 1px;
            transition: opacity 0.2s ease-in-out;
            opacity: 0;
            `;
    }

    // update the mutable popup attributes
    updatePopup() {
        const { style } = this.popup;
        style.fontFamily = options.font;
        style.fontSize = options.fontSize;
        style.fontWeight = options.fontWeight;
        style.fontStyle = options.italic ? "italic" : "";
    }

    /**
     * register or change the label of the menu command
     * @param {string} cap (intended caption to display on the menu command)
     * @param {bool} firstRun (we call this function at startup and every time we pause/unpause. we don't need to register a menu command if this is the startup call, since none exists yet)
     */
    register(cap, firstRun = false) {
        if (GM4) return; // don't register a menu command if the script manager is greasemonkey 4.0+ since the function doesn't exist
        if (!firstRun) GM_unregisterMenuCommand(this.caption); // this is how we switch the menu command from play to pause. we'd prefer to just have a single menu command and use a variable to determine its label and callback behavior, but the API doesn't support that afaik.
        // don't register the pause/unpause menu command if the site is currently disabled
        if (options[site]) {
            GM_registerMenuCommand(cap, this.toggle);
            this.caption = cap;
        }
    }

    // start listening to key events
    startCapturing() {
        win.addEventListener("keydown", this, true);
    }

    // stop listening to key events
    stopCapturing() {
        win.removeEventListener("keydown", this, true);
    }
}

// if using greasemonkey 4, remap the GM_* functions to GM.*
async function checkGM() {
    if (GM4) {
        GM_getValue = GM.getValue;
        GM_setValue = GM.setValue;
        GM_listValues = GM.listValues;
        GM_deleteValue = GM.deleteValue;
        GM_openInTab = GM.openInTab;
    }
}

// override API functions so we can animate the settings panel and auto-close it on save.
function extendGMC() {
    // support fancy animations
    GM_config.close = function close() {
        win.clearTimeout(this.fading);
        this.frame.setAttribute("closed", true);
        this.onClose(); //  Call the close() callback function
        this.isOpen = false;
        this.fading = win.setTimeout(() => {
            this.clearSheets("Marathon");
            // If frame is an iframe then remove it
            if (this.frame.contentDocument) {
                this.remove(this.frame);
                this.frame = null;
            } else {
                // else wipe its content
                this.frame.innerHTML = "";
                this.frame.style.display = "none";
            }
            // Null out all the fields so we don't leak memory
            const { fields } = this;
            for (const value of Object.values(fields)) {
                value.wrapper = null;
                value.node = null;
            }
        }, 500);
    };
    GM_config.open = function open() {
        win.clearTimeout(this.fading);
        this.frame.removeAttribute("closed");
        this.isOpen = true;
        Object.getPrototypeOf(this).open.call(this);
    };
    // override write function to semi-publicly memoize the error state.
    GM_config.write = function write(store, obj) {
        const values = {};
        const forgotten = {};
        if (!obj) {
            const { fields } = this;
            for (const [id, field] of Object.entries(fields)) {
                const value = field.toValue();
                if (field.save)
                    if (value != null) {
                        values[id] = value;
                        field.value = value;
                    } else {
                        this.error = true;
                        values[id] = field.value;
                    }
                else forgotten[id] = value;
            }
        }
        try {
            this.setValue(store || this.id, this.stringify(obj || values));
        } catch (e) {
            this.log("GM_config failed to save settings!");
        }
        return forgotten;
    };
    /**
     * remove all the stylesheets generated by GM_config. without this, GM_config keeps adding a new one every time you open it. we could resolve this oversight by overriding GM_config's open() method, but that's a lot of text to duplicate and this isn't expensive. also, deleting superfluous stuff is more satisfying than doing the proper thing and never creating it in the first place.
     * @param {string} sel (CSS selector; check each stylesheet for this string)
     */
    GM_config.clearSheets = (sel) => {
        for (const i of Array.from(methods.byTag("style", doc.head)))
            try {
                if (
                    i instanceof HTMLStyleElement &&
                    i.sheet.cssRules[0].selectorText &&
                    i.sheet.cssRules[0].selectorText.includes(sel)
                )
                    i.remove();
                // Amazon CSP blocks cross-origin use of method sheet.cssRules so the loop will interrupt on some unrelated stylesheet.
                // I'd use the optional chaining operator here but it's not enabled by default in chrome. So trycatch statement instead.
                // eslint-disable-next-line no-empty
            } catch (e) {}
    };
    /**
     * remove all the link elements generated by webfontloader.js. the loader has no logic to amend its existing stylesheets and will just keep adding more for every time you call it. since we call it every time the user changes the font settings, it makes sense to delete the previous ones before calling the load method.
     * @param {string} uri (url or part of url; check each link element's href attribute for this string)
     */
    GM_config.clearLinks = (uri) => {
        for (const i of Array.from(methods.byTag("link", doc.head)))
            if (i instanceof HTMLLinkElement && i.href.includes(uri)) i.remove();
    };
    /**
     * return true if any of the fields passed have values that deviate from their default values. we use this to avoid performing operations that are unnecessary when aspects of the user's config are unchanged.
     * @param {object} fields (an object whose properties are GM_config fields)
     */
    GM_config.checkNotDefault = (fields) =>
        !Object.values(fields).every((field) => field.value === field.default);
    // if webfont is enabled and any of the fields that affect webfont are non-default, (font, italic, fontWeight) then change the webfont config
    GM_config.updateWFConfig = function updateWFConfig() {
        if (options.webfont && this.checkNotDefault(this.webFontFields))
            WebFontConfig.google.families[1] = `${options.font}:${
                options.italic ? "ital," : ""
            }wght@1,${options.fontWeight}`;
    };
}

/**
 * set up the GM_config settings GUI
 */
async function initGMC() {
    await checkGM();
    const frame = doc.createElement("div");
    const sitesFieldLabel = methods.create(doc, "div", {
        class: "field_label",
        id: "Marathon_section_0_subheader_0",
    });
    sitesFieldLabel.innerHTML = "Run on: ";
    const resetBtn = doc.createElement("button");
    const supportBtn = doc.createElement("button");
    frame.style.display = "none";
    doc.body.appendChild(frame);
    frame.appendChild(sitesFieldLabel);
    frame.appendChild(resetBtn);
    frame.appendChild(supportBtn);
    resetBtn.addEventListener("click", () => GM_config.reset());
    supportBtn.addEventListener("click", () =>
        GM_openInTab("https://gf.qytechs.cn/scripts/420475-netflix-marathon-pausable")
    );
    GM_config.error = false; // this switch tells us if the user input an invalid value for a setting so we won't close the GUI when they try to save.
    extendGMC();
    // initialize the GUI
    GM_config.init({
        id: "Marathon",
        title: "Netflix Marathon Settings",
        fields: {
            amazon: {
                type: "checkbox",
                label: "Amazon",
                title: "Uncheck if you don't use Amazon Prime Video",
                section: "Main Settings",
                default: true,
            },
            netflix: {
                type: "checkbox",
                label: "Netflix",
                title: "Uncheck if you don't use Netflix",
                default: true,
            },
            disneyplus: {
                type: "checkbox",
                label: "Disney+",
                title: "Uncheck if you don't use Disney+",
                default: true,
            },
            rate: {
                label: "Interval Rate",
                title: "Time (in milliseconds) between checks for skip buttons",
                type: "int",
                size: 2,
                min: 50,
                max: 5000,
                default: 300,
            },
            promoted: {
                type: "checkbox",
                label: "Autoplay promoted videos",
                title: "After the final credits of a film or the last episode of a series, Netflix and Disney+ recommend a trending or similar movie/series. Check this if you want to automatically start playing the site's recommendation at the end of the credits",
                default: false,
            },
            code: {
                label: "Hotkey code",
                title: "Which keyboard key to use (click Support for a list of key codes)",
                type: "text",
                section: "Pause/Resume Hotkey",
                size: 6,
                default: "F7",
            },
            hotkey: {
                type: "checkbox",
                label: "Enable toggle hotkey",
                title: "Uncheck to disable the pause/resume shortcut",
                default: true,
            },
            ctrlKey: {
                type: "checkbox",
                label: "Ctrl key",
                title: "Set Ctrl as a modifier key for the shortcut",
                default: true,
            },
            altKey: {
                type: "checkbox",
                label: "Alt key",
                title: "Set Alt as a modifier key for the shortcut",
                default: false,
            },
            shiftKey: {
                type: "checkbox",
                label: "Shift key",
                title: "Set Shift as a modifier key for the shortcut",
                default: false,
            },
            metaKey: {
                type: "checkbox",
                label: "Meta key",
                title: "Set Meta as a modifier key for the shortcut",
                default: false,
            },
            code2: {
                label: "Hotkey code",
                title: "Which keyboard key to use (click Support for a list of key codes)",
                type: "text",
                section: "Settings Hotkey",
                size: 6,
                default: "KeyN",
            },
            hotkey2: {
                type: "checkbox",
                label: "Enable settings hotkey",
                title: "Uncheck to disable the keyboard shortcut",
                default: true,
            },
            ctrlKey2: {
                type: "checkbox",
                label: "Ctrl key",
                title: "Set Ctrl as a modifier key for the shortcut",
                default: false,
            },
            altKey2: {
                type: "checkbox",
                label: "Alt key",
                title: "Set Alt as a modifier key for the shortcut",
                default: true,
            },
            shiftKey2: {
                type: "checkbox",
                label: "Shift key",
                title: "Set Shift as a modifier key for the shortcut",
                default: false,
            },
            metaKey2: {
                type: "checkbox",
                label: "Meta key",
                title: "Set Meta as a modifier key for the shortcut",
                default: false,
            },
            pop: {
                type: "checkbox",
                label: "Enable popup",
                title: "Uncheck to disable the Paused/Resumed popups",
                section: "Popup Settings",
                default: true,
            },
            popDur: {
                label: "Popup duration",
                title: "How long (in milliseconds) the popup should stay open before fading away",
                type: "int",
                size: 4,
                min: 500,
                max: 50000,
                default: 3000,
            },
            webfont: {
                type: "checkbox",
                label: "Use Google Fonts",
                title: "If the font you want is not locally installed, this must be checked",
                default: true,
            },
            font: {
                label: "Popup font",
                title: "Which font to use for the Paused/Resumed popups",
                type: "text",
                size: 12,
                default: "Source Sans Pro",
            },
            fontSizeInt: {
                label: "Font size (px)",
                title: "How big the Paused/Resumed popups should be",
                type: "int",
                size: 1,
                min: 6,
                max: 560,
                default: 24,
            },
            fontWeight: {
                label: "Font weight",
                title: "Boldness of the popup text, measured in multiples of 100 from 100-900",
                type: "select",
                options: ["100", "200", "300", "400", "500", "600", "700", "800", "900"],
                default: 300,
            },
            italic: {
                type: "checkbox",
                label: "Italic",
                title: "Check if you want the popup text to be italic",
                default: false,
            },
        },
        events: {
            init() {
                // determine if user has any orphaned script settings
                const migrateKeys = GM_listValues().filter((key) => key !== "Marathon");
                const f = this.fields;
                // all the fields that affect the pause/resume hotkey
                this.hotkeyFields = {
                    code: f.code,
                    hotkey: f.hotkey,
                    ctrlKey: f.ctrlKey,
                    altKey: f.altKey,
                    shiftKey: f.shiftKey,
                    metaKey: f.metaKey,
                    code2: f.code2,
                    hotkey2: f.hotkey2,
                    ctrlKey2: f.ctrlKey2,
                    altKey2: f.altKey2,
                    shiftKey2: f.shiftKey2,
                    metaKey2: f.metaKey2,
                };
                // all the fields that affect the popup appearance/existence
                this.popupFields = {
                    pop: f.pop,
                    popDur: f.popDur,
                    webfont: f.webfont,
                    font: f.font,
                    fontSizeInt: f.fontSizeInt,
                    fontWeight: f.fontWeight,
                    italic: f.italic,
                };
                // all the fields that necessitate loading a new font stylesheet with webfont.
                this.webFontFields = {
                    font: f.font,
                    fontWeight: f.fontWeight,
                    italic: f.italic,
                };
                // this exists to migrate user settings from the old system (multiple objects) to the new system (one JSON object with multiple keys)
                if (migrateKeys.length)
                    for (const key of migrateKeys) {
                        const oldVal = GM_getValue(key);
                        // fontSize used to be a string setting, now it's an integer setting fontSizeInt. need to convert it first
                        if (key === "fontSize" && typeof oldVal === "string") {
                            const newVal = Number(oldVal.match(/\d+/g)[0]);
                            this.set("fontSizeInt", newVal);
                        } else this.set(key, oldVal);

                        GM_deleteValue(key); // get rid of the old setting so we don't have to do this again.
                    }
                this.save(); // we need this to save the default values on first load
                // for all addons except greasemonkey 4, we can add a menu command
                if (!GM4)
                    GM_registerMenuCommand("Open Settings", () => {
                        if (!this.isOpen) this.open();
                    });
                // memoize the settings
                settings();
            },
            save() {
                if (this.isOpen) {
                    // don't do anything until the user fixes their invalid input
                    if (this.error) return (this.error = false);
                    const f = this.fields;
                    let message = "";
                    let hotkeyMsg = false;
                    let doResetPopup = false;
                    let doReloadWF = false;
                    // close the settings menu upon save (provided none of the inputs is invalid)
                    this.close();
                    // handle changes to any hotkey-related settings
                    for (const [key, field] of Object.entries(this.hotkeyFields)) {
                        const tempKey = field.value;
                        // if the memoized setting doesn't match the new value...
                        if (options[key] !== tempKey) {
                            options[key] = tempKey; // update it
                            if (key === "hotkey" || key === "hotkey2")
                                // if the enable hotkey setting was changed, either stop or start the keydown listener
                                options.hotkey || options.hotkey2 // if either of these settings is true, we need the event listener
                                    ? marathon.startCapturing()
                                    : marathon.stopCapturing();
                            // if both are false then there's no need to listen to keydown at all
                            hotkeyMsg = true;
                        }
                    }
                    if (hotkeyMsg) message += "Hotkeys"; // tell popup to open and announce the successful settings update
                    // handle changes to popup settings
                    for (const [key, field] of Object.entries(this.popupFields)) {
                        const tempKey = field.value;
                        if (options[key] !== tempKey) {
                            options[key] = tempKey; // same pattern as for the hotkey fields, but with some more bespoke behavior below
                            switch (key) {
                                case "pop":
                                    return (this.error = false); // do nothing special if the popup was enabled/disabled since the toggle already checks the option
                                case "webfont":
                                    if (!tempKey) WebFontConfig.google.families.splice(1, 1); // if webfont was disabled, then remove the user-defined font from the webfont config
                                // break omitted
                                case "font":
                                case "fontWeight":
                                case "italic":
                                    doReloadWF = true; // if font, fontWeight, or italic were changed, we need to use webfontloader again
                                    break;
                                case "fontSizeInt":
                                    options.fontSize = `${tempKey}px`; // if font size was changed, update the internal string version
                                    break;
                                default:
                            }
                            doResetPopup = true;
                        }
                    }
                    // if anything was changed, tell popup to announce it
                    if (doResetPopup) {
                        if (doReloadWF) {
                            this.clearLinks("fonts.googleapis.com"); // clear old webfont sheets
                            this.updateWFConfig(); // update WebFontConfig
                            WebFont.load(WebFontConfig); // load new font sheet
                        }
                        if (options.pop) {
                            marathon.updatePopup(); // if the script started with popups disabled, then the styles won't exist yet, so load them.
                            if (message) message += " & "; // if we already set message to Hotkey...
                            message += "Popup"; // set it to Hotkey & Popup
                        }
                    }

                    // handle changes to rate and site settings
                    const newInt = f.rate.value;
                    if (options.rate !== newInt) {
                        options.rate = newInt;
                        marathon.pause(); // stop the current interval
                        marathon.int = newInt; // update the rate
                        if (options[site]) marathon.resume(); // if the site we're currently on is enabled, start the interval with the new rate
                        if (message.includes("&")) message = "Settings";
                        // if we already set it to Hotkey & Popup then reset it to something general so it's not so long
                        else {
                            if (message) message += " & "; // otherwise if it's set to either Hotkey *or* Popup, set it to e.g. Hotkey & Interval
                            message += "Interval"; // otherwise just set it to Interval
                        }
                    }
                    if (
                        options.netflix !== f.netflix.value ||
                        options.amazon !== f.amazon.value ||
                        options.disneyplus !== f.disneyplus.value ||
                        options.promoted !== f.promoted.value
                    ) {
                        // if the memoized setting for the current site doesn't match the new setting for that site...
                        if (options[site] !== f[site].value) {
                            options[site] = f[site].value; // make them match...
                            f[site].value // and stop or start the interval accordingly
                                ? marathon.resume()
                                : marathon.pause();
                        }
                        options.promoted = f.promoted.value;
                        // if we already changed other types of settings then set the message to something general
                        if (message) message = "Settings";
                        else message = "Site Settings"; // otherwise make it specific to site settings.
                    }
                    if (message) marathon.openPopup(`Updated ${message}`); // finally open a popup with whatever message we gave.
                }
                return (this.error = false);
            },
            open() {
                // add a label to the amazon/netflix/disney+ checkboxes
                methods.byID("Marathon_section_header_0").after(sitesFieldLabel);
                // add a support button, make the reset link an actual button. we could do this by editing the prototype but again, it'd be a lot of duplicate code.
                const resetLink = methods.byID("Marathon_resetLink"); // the ugly reset link that comes with GM_config
                methods.maybeSetAttributes(resetBtn, {
                    title: resetLink.title,
                    class: resetLink.parentElement.className,
                });
                resetBtn.textContent = resetLink.textContent;
                resetLink.parentElement.replaceWith(resetBtn); // replace the link with the button
                methods.byID("Marathon_saveBtn").after(resetBtn); // move it next to the save button
                // give the support button a localized tooltip and label since it's the one someone's most likely to need if they don't speak english.
                methods.maybeSetAttributes(supportBtn, {
                    title: l10n.title,
                    class: "saveclose_buttons",
                    id: "Marathon_supportBtn",
                });
                supportBtn.textContent = l10n.text;
                methods.byID("Marathon_closeBtn").after(supportBtn); // move it to the end.
                const firstField = methods.qry(`.config_var [id^="Marathon_field_"]`, frame);
                if (firstField) firstField.focus();
            },
            close() {
                let blurTo;
                switch (site) {
                    case "netflix": {
                        const mountPoint = methods.byID("appMountPoint");
                        blurTo = methods.qry(`[tabindex]`, mountPoint) || mountPoint;
                        break;
                    }
                    case "amazon":
                        blurTo = methods.qry(".webPlayerSDKUiContainer");
                        break;
                    case "disneyplus":
                        blurTo = methods.qry(".btm-media-client-element");
                        break;
                    default:
                        return;
                }
                blurTo.focus();
            },
        },
        frame, // using an in-content element has its problems e.g. we're affected by amazon's god-awful stylesheets, but using an iframe makes animation a lot more clunky and i want the panel to be kinda spry and light
        css: `#Marathon{display:block!important;position:fixed!important;z-index:2147483646!important;inset:unset!important;top:50%!important;left:0!important;background-color:hsla(0,0%,5.1%,.91);background-image:url("https://cdn.jsdelivr.net/gh/aminomancer/Netflix-Marathon-Pausable@latest/texture/noise-512x512.png");background-repeat:repeat;background-size:auto;background-attachment:local;-webkit-backdrop-filter:blur(7px);backdrop-filter:blur(7px);border:none!important;color:hsla(0,0%,97%,.95);max-width:-webkit-min-content!important;max-width:-moz-min-content!important;max-width:min-content!important;height:-webkit-min-content!important;height:-moz-min-content!important;height:min-content!important;border-radius:5px;padding:10px!important;transform:translate(50%,-60%);font-size:14px;line-height:1.2;transition:.2s ease-in-out opacity;}#Marathon[closed]{opacity:0!important;transition:.5s ease-in-out opacity;}#Marathon *{font-family:Source Sans Pro;font-weight:300;}#Marathon_wrapper{display:flex;flex-direction:column;align-content:center;}#Marathon_header{font-size:2em!important;white-space:nowrap;padding-inline:6px;}#Marathon .section_header_holder{display:flex;flex-flow:row wrap;gap:6px 8px;padding:6px 4px 0;margin-top:8px;border-top:1px solid hsla(0,0%,100%,.1);}#Marathon .section_header{font-size:1.25em!important;background:none!important;border:none!important;text-align:left!important;flex-basis:100%;margin-inline:-3px;}#Marathon .config_var{margin:0!important;column-gap:6px;display:flex;flex-direction:row;align-items:center;line-height:normal;flex-grow:1;}#Marathon:is(button,input,optgroup,select,textarea){font:inherit;margin:0;}#Marathon button{text-align:center;}#Marathon input[type="text"]{-webkit-appearance:none;-moz-appearance:none;appearance:none;color:inherit;background:hsla(0,0%,25%,50%)!important;border:none;border-radius:3px;padding:1px 4px;flex-grow:1;height:unset;box-shadow:unset!important;outline:unset;box-sizing:initial!important;margin:0!important;font-size:14px!important;}#Marathon input[type="text"]:focus{background-color:hsla(0,0%,25%,70%)!important;color:white!important;}#Marathon input[type="checkbox"]{min-width:14px;min-height:14px;margin-inline:0;border-radius:2.5px;position:static;box-sizing:border-box;}#Marathon:is(select,button,textarea):focus-visible,#Marathon input:not([type="file"],[type="image"]):focus-visible,#Marathon:is(button,select,input:is([type="checkbox"],[type="color"],[type="radio"])):-moz-focusring{outline-style:auto!important;box-shadow:none!important;}#Marathon_section_0{gap:6px 12px;}#Marathon_section_0_subheader_0{flex-grow:1;}#Marathon_buttons_holder{display:flex;flex-flow:row;gap:6px;margin-top:6px;align-items:center;border-top:1px solid hsla(0,0%,100%,.1);color:inherit!important;}#Marathon .saveclose_buttons,#Marathon .reset_holder{margin:6px 0 0 0;padding:2px 12px;color:inherit;background:hsla(0,0%,25%,50%);border:none!important;border-radius:3px;padding-inline:4px;font-size:15px;padding-block:2px;flex-grow:1;white-space:nowrap;}#Marathon .saveclose_buttons:hover,#Marathon .reset_holder:hover{background-color:hsla(0,0%,25%,70%)!important;color:white!important;}#Marathon_saveBtn{padding-inline:16px 2px!important;background:hsla(0,0%,25%,50%) url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='16' height='16'><path fill='hsla(0,0%,97%,.95)' d='M6 14a1 1 0 01-.707-.293l-3-3a1 1 0 011.414-1.414l2.157 2.157 6.316-9.023a1 1 0 011.639 1.146l-7 10a1 1 0 01-.732.427A.863.863 0 016 14z'/></svg>") 3.8px 48%/12.5px no-repeat!important;}#Marathon .reset_holder{padding-inline:16px 2px!important;background:hsla(0,0%,25%,50%) url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' fill='hsla(0,0%,97%,.95)' height='16' width='16'><path d='M1 1a1 1 0 011 1v2.4A7 7 0 118 15a7 7 0 01-4.9-2 1 1 0 011.4-1.5 5 5 0 10-1-5.5H6a1 1 0 010 2H1a1 1 0 01-1-1V2a1 1 0 011-1z'/></svg>") 4.5px 50%/11px no-repeat!important;}#Marathon .reset{color:inherit!important;font-size:inherit!important;}#Marathon .field_label{font-size:12px;font-weight:normal!important;margin:0!important;white-space:nowrap;padding:unset!important;}#Marathon select{-webkit-appearance:none;-moz-appearance:none;appearance:none;color:inherit;border:none;border-radius:3px;padding-inline:2px 13px;background:hsla(0,0%,25%,50%) url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' fill='hsla(0,0%,97%,.95)' height='24' viewBox='0 0 24 24' width='24'><path d='M8.12 9.29L12 13.17l3.88-3.88c.39-.39 1.02-.39 1.41 0 .39.39.39 1.02 0 1.41l-4.59 4.59c-.39.39-1.02.39-1.41 0L6.7 10.7c-.39-.39-.39-1.02 0-1.41.39-.38 1.03-.39 1.42 0z'/></svg>") 100% 66%/18px no-repeat!important;}#Marathon option{-webkit-appearance:none;-moz-appearance:none;appearance:none;color:inherit;background:hsl(0,1%,17%)!important;border:none;}#Marathon_pop_var,#Marathon_font_var{flex-basis:100%;}`,
    });
}

// load webfontloader and create the base config (to be changed by GM_config)
function attachWebFont() {
    const wf = doc.createElement("script");
    const first = doc.scripts[0];
    WebFontConfig = {
        classes: false, // don't bother changing the DOM at all, we aren't listening for it
        events: false, // no need for events, not worth the execution
        google: {
            families: ["Source Sans Pro:wght@1,300"], // default font and settings font
            display: "swap", // not really necessary since the popup doesn't appear until you press a button. but whatever
        },
    };
    GM_config.updateWFConfig(); // parse user-defined font settings, if any
    wf.src = "https://cdn.jsdelivr.net/npm/webfontloader@latest/webfontloader.js";
    wf.async = true; // don't block the rest of the page for this
    first.parentNode.insertBefore(wf, first);
}

// after getting settings from *monkey storage, memoize their values in options.
async function settings() {
    for (const [key, field] of Object.entries(GM_config.fields)) options[key] = field.value;
    options.fontSize = `${options.fontSizeInt}px`;
}

async function start() {
    await initGMC(); // wait for GM_config
    marathon = new Controller(methods, options.rate); // create the interval controller, event listeners, etc.
    attachWebFont(); // load the font sheet
}

start();

QingJ © 2025

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