YouTube 播放速度记忆

自动切换到你预先设定的播放速度。

目前为 2024-08-15 提交的版本。查看 最新版本

// ==UserScript==
// @name                Youtube Remember Speed
// @name:zh-TW          YouTube 播放速度記憶
// @name:zh-CN          YouTube 播放速度记忆
// @name:ja             YouTube 再生速度メモリー
// @icon                
// @author              ElectroKnight22
// @namespace           electroknight22_youtube_remember_playback_rate_namespace
// @version             1.0.1
// @match               *://www.youtube.com/*
// @exclude             *://www.youtube.com/live_chat*
// @grant               GM.getValue
// @grant               GM.setValue
// @grant               GM.deleteValue
// @grant               GM.listValues
// @grant               GM_registerMenuCommand
// @grant               GM_unregisterMenuCommand
// @license             MIT
// @description         Automcatically switches to your pre-selected speed.
// @description:zh-TW   自動切換到你預先設定的播放速度。
// @description:zh-CN   自动切换到你预先设定的播放速度。
// @description:ja      自動的に設定した再生速度に替わります。
// ==/UserScript==

/*jshint esversion: 11 */

(function() {
    "use strict";

    const DEBUG = false;

    const DEFAULT_SETTINGS = {
        targetSpeed: 1
    };

    const speeds = [0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];

    let userSettings = { ...DEFAULT_SETTINGS };
    let menuCommandIds = [];

    let doc = document, win = window;

    // --------------------
    // --- FUNCTIONS ------
    // --------------------

    function debugLog(message, shouldShow = true) {
        if (DEBUG && shouldShow) {
            console.log("YTRS DEBUG | " + message);
        }
    }

    // --------------------
    // Attempt to set the video resolution to target quality or the next best quality
    function setSpeed(targetSpeed) {

        let ytPlayer = doc.getElementById("movie_player") || doc.getElementsByClassName("html5-video-player")[0];

        if (!isValidVideo(ytPlayer)) return;

        ytPlayer.setPlaybackRate(targetSpeed);
        debugLog("Trying to set speed to: " + targetSpeed + "x");

    }

    function isValidVideo(ytPlayer, force) {

        if (!ytPlayer?.getAvailableQualityLabels()[0]) {
            debugLog("Video data missing");
            return false;
        }

        if (win.location.href.startsWith("https://www.youtube.com/shorts/")) {
            debugLog("Skipping Youtube Shorts");
            return false;
        }

        return true;
    }

    // --------------------
    // Functions for the speed selection menu

    function createSpeedMenu() {
        GM_registerMenuCommand("Set Speed (show/hide)", () => {
            menuCommandIds.length ? removeSpeedMenuItems() : showSpeedMenuItems();
        }, {
            autoClose: false
        });
    }

    function showSpeedMenuItems() {
        removeSpeedMenuItems();
        speeds.forEach((speed) => {
            let speedText = speed + "x";
            if (speed === userSettings.targetSpeed) {
                speedText += " (selected)";
            }
            let menuCommandId = GM_registerMenuCommand(speedText, () => {
                setSelectedSpeed(speed);
            }, {
                autoClose: false,
            });
            menuCommandIds.push(menuCommandId);
        });
    }

    function removeSpeedMenuItems() {
        while (menuCommandIds.length) {
            GM_unregisterMenuCommand(menuCommandIds.pop());
        }
    }

    function setSelectedSpeed(speed) {
        if (userSettings.targetSpeed == speed) return;
        userSettings.targetSpeed = speed;
        GM.setValue('targetSpeed', speed);
        removeSpeedMenuItems();
        showSpeedMenuItems();
        setSpeed(speed);
    }

    // --------------------
    // Sync settings with locally stored values
    async function applySettings() {
        try {
            // Get all keys from GM
            const storedValues = await GM.listValues();

            // Write any missing key-value pairs from DEFAULT_SETTINGS to GM
            await Promise.all(Object.entries(DEFAULT_SETTINGS).map(async ([key, value]) => {
                if (!storedValues.includes(key)) {
                    await GM.setValue(key, value);
                }
            }));

            // Delete any extra keys in GM that are not in DEFAULT_SETTINGS
            await Promise.all(storedValues.map(async key => {
                if (!(key in DEFAULT_SETTINGS)) {
                    await GM.deleteValue(key);
                }
            }));

            // Retrieve and update user settings from GM
            await Promise.all(
                storedValues.map(key => GM.getValue(key).then(value => [key, value]))
            ).then(keyValuePairs => keyValuePairs.forEach(([newKey, newValue]) => {
                userSettings[newKey] = newValue;
            }));

            debugLog(Object.entries(userSettings).map(([key, value]) => key + ": " + value).join(", "));
        } catch (error) {
            debugLog("Error when applying settings: " + error.message);
        }
    }

    // --------------------
    // Main function
    function main() {
        if (win.self == win.top) { createSpeedMenu(); }
        setSpeed(userSettings.targetSpeed);
        win.addEventListener("loadstart", () => { setSpeed(userSettings.targetSpeed); }, true);
    }

    // --------------------
    // Entry Point
    applySettings().then(main);
})();