// ==UserScript== // @name Youtube HD Premium // @icon  // @author ElectroKnight22 // @namespace electroknight22_youtube_hd_namespace // @version 2024.07.02 // @match *://www.youtube.com/* // @grant GM.getValue // @grant GM.setValue // @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"; // --- SETTINGS ------- // !!! PLEASE NOTE: // !!! Having the incorrect "overwriteStoredSettings" value maybe cause your settings to not save. let settings = { // true = use the setting here. Use this when changing settings here. // false = use stored setting. Use this when changing settings from the "Storage" tab. overwriteStoredSettings: false, // true = use the setting you last chose on YouTube. // false = ignore YouTube settings. importYoutubeSettings: false, // The target resolution. If not available, the next best available resolution will be used. targetRes: "hd2160" // Choices for targetRes are currently: // "highres" >= ( 8K / 4320p / QUHD ) // "hd2880" = ( 5K / 2880p / UHD+ ) // "hd2160" = ( 4K / 2160p / UHD ) // "hd1440" = ( 1440p / QHD ) // "hd1080" = ( 1080p / FHD ) // "hd720" = ( 720p / HD ) // "large" = ( 480p ) // "medium" = ( 360p ) // "small" = ( 240p ) // "tiny" = ( 144p ) // "auto" = ( auto ) }; // -------------------- // --- GLOBALS -------- // -------------------- const DEBUG = false; 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 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) { let ytPlayer = doc.getElementById("movie_player") || doc.getElementsByClassName("html5-video-player")[0]; if (!isValidVideo(ytPlayer)) return; vidId = ytPlayer.getVideoData().video_id; let localItem = localStorage.getItem("yt-player-quality"); if (localItem && settings.importYoutubeSettings) { let youtubeSettings = fetchLocalSettings(localItem); if (qualityLevels[youtubeSettings.quality]) { target = qualityLevels[youtubeSettings.quality]; } } 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) { if (!ytPlayer?.getAvailableQualityLabels()[0]) { debugLog("Video data missing"); return false; } if (vidId == ytPlayer.getVideoData().video_id) { debugLog("Duplicate load"); return false; } return true; } function main() { let target = settings.targetRes.toLowerCase(); if (target != 'auto') { setResolution(target); win.addEventListener("loadstart", () => { setResolution(target); }, true); } } function fetchLocalSettings(localItem) { let quality = 0; let previousQuality = 0; let expiration = 0; let creation = 0; if (localItem) { localItem = JSON.parse(localItem); let localData = JSON.parse(localItem.data); quality = localData.quality; previousQuality = localData.previousQuality; expiration = localItem.expiration; creation = localItem.creation; } return { quality, previousQuality, expiration, creation }; } async function applySettings() { if (typeof GM != 'undefined' && GM.getValue && GM.setValue && GM.deleteValue && GM.listValues) { if (settings.overwriteStoredSettings) { // Clear all values from GM const keys = await GM.listValues(); await Promise.all(keys.map(key => GM.deleteValue(key))); // Write each key-value pair from settings await Promise.all(Object.entries(settings).map(([k, v]) => GM.setValue(k, v))); } else { // Get all keys from GM const gmKeys = await GM.listValues(); // Write any missing key-value pairs from settings to GM await Promise.all(Object.entries(settings).map(async ([k, v]) => { if (!gmKeys.includes(k)) { await GM.setValue(k, v); } })); // Delete any extra keys in GM that are not in settings await Promise.all(gmKeys.map(async key => { if (!(key in settings)) { await GM.deleteValue(key); } })); // Update the value for .overwriteStoredSettings in GM await GM.setValue('overwriteStoredSettings', settings.overwriteStoredSettings); // Retrieve and update settings from GM await Promise.all( gmKeys.map(k => GM.getValue(k).then(v => [k, v])) ).then(c => c.forEach(([nk, nv]) => { if (settings[nk] !== null && nk !== "overwriteStoredSettings") { settings[nk] = nv; } })); } debugLog(Object.entries(settings).map(([k, v]) => k + ": " + v).join(", ")); } } applySettings().then(main); })();