// ==UserScript==
// @name Youtube HD Premium
// @icon 
// @author ElectroKnight22
// @namespace electroknight22_youtube_hd_namespace
// @version 2024.07.07.2
// @match *://www.youtube.com/*
// @exclude *://www.youtube.com/live_chat*
// @grant GM.getValue
// @grant GM.setValue
// @grant GM.listValues
// @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のビットレートを優先的に選択します。
// ==/UserScript==
/*jshint esversion: 11 */
(function() {
"use strict";
const DEBUG = true;
const DEFAULT_SETTINGS = {
targetResolution: "hd2160"
};
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 availableQualities = ytPlayer.getAvailableQualityLevels();
target = findNextAvailableQuality(target, availableQualities);
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");
vidId = null;
return false;
}
if (window.location.href.startsWith("https://www.youtube.com/shorts/")) {
debugLog("Skipping Youtube Shorts");
vidId = null;
return false;
}
if (vidId == ytPlayer.getVideoData().video_id && !force) {
debugLog("Duplicate load");
return false;
}
return true;
}
function findNextAvailableQuality(target, availableQualities) {
const available = availableQualities.map(q => ({ quality: q, value: quality[q] }));
const targetValue = quality[target];
const smallerOrEqualQualities = available.filter(q => q.value <= targetValue);
smallerOrEqualQualities.sort((a, b) => b.value - a.value);
return smallerOrEqualQualities.length > 0 ? smallerOrEqualQualities[0].quality : 'auto';
}
// --------------------
// Functions for the quality selection menu
function createQualityMenu() {
const menu_command_id_0 = GM_registerMenuCommand("Set Preferred Quality (show/hide)", 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() {
if (win.self == win.top) { createQualityMenu(); }
setResolution(userSettings.targetResolution);
win.addEventListener("loadstart", () => { setResolution(userSettings.targetResolution); }, true);
}
// --------------------
// Entry Point
applySettings().then(main);
})();