// ==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 自動的に設定した再生速度に替わります。 // @downloadURL none // ==/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); })();