// ==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のビットレートを優先的に選択します。
// ==/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);
})();