// ==UserScript== // @name Youtube HD Premium // @icon  // @author ElectroKnight22 // @namespace electroknight22_youtube_hd_namespace // @version 2024.07.06 // @match *://www.youtube.com/* // @exclude *://www.youtube.com/live_chat* // @grant GM.getValue // @grant GM.setValue // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @license MIT // @description    Automcatically switches to your pre-selected resolution. Enables premium when possible. // @description:zh-TW 自動切換到你預先設定的畫質。會優先使用Premium位元率。 // @description:zh-CN 自动切换到你预先设定的画质。会优先使用Premium比特率。 // @description:ja 自動的に設定した画質に替わります。Premiumのビットレートを優先的に選択します。 // @downloadURL none // ==/UserScript== /*jshint esversion: 11 */ (function() { "use strict"; const DEFAULT_SETTINGS = { targetResolution: "hd2160" }; const DEBUG = true; const resolutions = ['highres', 'hd2880', 'hd2160', 'hd1440', 'hd1080', 'hd720', 'large', 'medium', 'small', 'tiny', 'auto']; const quality = { highres: 4320, hd2880: 2880, hd2160: 2160, hd1440: 1440, hd1080: 1080, hd720: 720, large: 480, medium: 360, small: 240, tiny: 144, auto: 0 }; const qualityLevels = Object.fromEntries( Object.entries(quality).map(([key, value]) => [value, key]) ); let userSettings = { ...DEFAULT_SETTINGS }; let menuCommandIds = []; let doc = document, win = window; let vidId = null; // -------------------- // --- FUNCTIONS ------ // -------------------- function debugLog(message, shouldShow = true) { if (DEBUG && shouldShow) { console.log("YTHD DEBUG | " + message); } } // -------------------- // Attempt to set the video resolution to target quality or the next best quality function setResolution(target, force = false) { if (target == 'auto') return; let ytPlayer = doc.getElementById("movie_player") || doc.getElementsByClassName("html5-video-player")[0]; if (!isValidVideo(ytPlayer, force)) return; vidId = ytPlayer.getVideoData().video_id; let localItem = null; try { localStorage.getItem("yt-player-quality"); } catch { debugLog("Fetching last used quality failed catastrophically. Likely the website is not YouTube. If website is YouTube then YouTube changed something."); } let limitList = ytPlayer.getAvailableQualityLevels(); let limit = limitList[0]; if (quality[target] > quality[limit]) { target = limit; } let premiumIndicator = "Premium"; let premiumData = ytPlayer.getAvailableQualityData().find(q => q.quality == target && q.qualityLabel.includes(premiumIndicator) && q.isPlayable); ytPlayer.setPlaybackQualityRange(target, target, premiumData?.formatId); debugLog("Set quality to: " + target + (premiumData ? " Premium" : "")); if (localItem){ localStorage.setItem("yt-player-quality",localItem); } else { localStorage.removeItem("yt-player-quality"); } } function isValidVideo(ytPlayer, force) { if (!ytPlayer?.getAvailableQualityLabels()[0]) { debugLog("Video data missing"); return false; } if (vidId == ytPlayer.getVideoData().video_id && !force) { debugLog("Duplicate load"); return false; } return true; } // -------------------- // Functions for the quality selection menu function createQualityMenu() { const menu_command_id_0 = GM_registerMenuCommand("Set Preferred Quality", function(MouseEvent) { menuCommandIds[0] ? removeQualityMenuItems() : showQualityMenuItems(); }, { autoClose: false }); } function showQualityMenuItems() { removeQualityMenuItems(); resolutions.forEach((resolution) => { let qualityText = (resolution === 'auto') ? 'auto' : quality[resolution] + "p"; if (resolution === userSettings.targetResolution) { qualityText += " (selected)"; } let menuCommandId = GM_registerMenuCommand(qualityText, function() { setSelectedResolution(resolution); }, { autoClose: false, }); menuCommandIds.push(menuCommandId); }); } function removeQualityMenuItems() { menuCommandIds.forEach((menuCommandId) => { GM_unregisterMenuCommand(menuCommandId); }); menuCommandIds = []; } function setSelectedResolution(resolution) { if (userSettings.targetResolution == resolution) return; userSettings.targetResolution = resolution; GM.setValue('targetResolution', resolution); removeQualityMenuItems(); showQualityMenuItems(); debugLog(resolution); setResolution(resolution, true); } // -------------------- // Sync settings with locally storaged values async function applySettings() { try { if (typeof GM != 'undefined' && GM.getValue && GM.setValue && GM.deleteValue && GM.listValues) { // Get all keys from GM const gmKeys = await GM.listValues(); // Write any missing key-value pairs from DEFAULT_SETTINGS to GM await Promise.all(Object.entries(DEFAULT_SETTINGS).map(async ([k, v]) => { if (!gmKeys.includes(k)) { await GM.setValue(k, v); } })); // Delete any extra keys in GM that are not in DEFAULT_SETTINGS await Promise.all(gmKeys.map(async key => { if (!(key in DEFAULT_SETTINGS)) { await GM.deleteValue(key); } })); // Retrieve and update user settings from GM await Promise.all( gmKeys.map(k => GM.getValue(k).then(v => [k, v])) ).then(c => c.forEach(([nk, nv]) => { userSettings[nk] = nv; })); debugLog(Object.entries(userSettings).map(([k, v]) => k + ": " + v).join(", ")); } } catch (error) { debugLog("Error in applySettings: " + error.message); } } // -------------------- // Main function function main() { createQualityMenu(); setResolution(userSettings.targetResolution); win.addEventListener("loadstart", () => { setResolution(userSettings.targetResolution); }, true); } // -------------------- // Entry Point applySettings().then(main); })();