diff --git a/src/config.js b/src/config.js index 5a318aa..137c8f2 100644 --- a/src/config.js +++ b/src/config.js @@ -2,6 +2,7 @@ const CONFIG_KEY = 'ytaf-configuration'; const configOptions = new Map([ ['enableAdBlock', { default: true, desc: 'Enable ad blocking' }], + ['upgradeThumbnails', { default: false, desc: 'Upgrade thumbnail quality' }], ['removeShorts', { default: true, desc: 'Remove Shorts from subscriptions' }], ['enableSponsorBlock', { default: true, desc: 'Enable SponsorBlock' }], [ diff --git a/src/thumbnail-quality.ts b/src/thumbnail-quality.ts new file mode 100644 index 0000000..12d4bc2 --- /dev/null +++ b/src/thumbnail-quality.ts @@ -0,0 +1,150 @@ +const webpTestImgs = { + lossy: 'UklGRiIAAABXRUJQVlA4IBYAAAAwAQCdASoBAAEADsD+JaQAA3AAAAAA', + lossless: 'UklGRhoAAABXRUJQVlA4TA0AAAAvAAAAEAcQERGIiP4HAA==', + alpha: + 'UklGRkoAAABXRUJQVlA4WAoAAAAQAAAAAAAAAAAAQUxQSAwAAAARBxAR/Q9ERP8DAABWUDggGAAAABQBAJ0BKgEAAQAAAP4AAA3AAP7mtQAAAA==', + animation: + 'UklGRlIAAABXRUJQVlA4WAoAAAASAAAAAAAAAAAAQU5JTQYAAAD/////AABBTk1GJgAAAAAAAAAAAAAAAAAAAGQAAABWUDhMDQAAAC8AAAAQBxAREYiI/gcA' +} as const; + +function checkWebpFeature( + feature: keyof typeof webpTestImgs, + callback: ( + featureName: keyof typeof webpTestImgs, + isSupported: boolean + ) => void +) { + const img = new Image(); + img.onload = function () { + const result = img.width > 0 && img.height > 0; + callback(feature, result); + }; + + img.onerror = function () { + callback(feature, false); + }; + + img.src = 'data:image/webp;base64,' + webpTestImgs[feature]; +} + +let webpSupported = false; +checkWebpFeature('lossy', (_, support) => { + webpSupported = support; +}); + +function rewriteURL(url: URL) { + const YT_THUMBNAIL_PATHNAME_REGEX = + /vi(?:_webp)?(\/.*?\/)([a-z0-9]+?)(_\w*?)?\.[a-z]+$/g; + + const YT_TARGET_THUMBNAIL_NAMES = [ + 'sddefault', + 'hqdefault', + 'mqdefault', + 'default' + ] as const; + + const isABTest = url.hostname.match(/^i\d/) !== null; + // Don't know how to handle A/B test thumbnails so we don't upgrade them. + if (isABTest) return null; + + const replacementPathname = url.pathname.replace( + YT_THUMBNAIL_PATHNAME_REGEX, + (match, p1, p2, p3) => { + if (!YT_TARGET_THUMBNAIL_NAMES.includes(p2)) return match; // Only rewrite regular thumbnail URLs. Not shorts, etc. + return `${webpSupported ? 'vi_webp' : 'vi'}${p1}sddefault${p3 ?? ''}.${webpSupported ? 'webp' : 'jpg'}`; + } + ); + if (url.pathname === replacementPathname) + // pathname not changed because not a regular thumbnail or already upgraded. + return null; + + url = new URL(url); + + url.pathname = replacementPathname; + url.search = ''; + + return url; +} + +function parseCSSUrl(value: string) { + return new URL(value.slice(4, -1).replace(/["']/g, '')); +} + +async function upgradeBgImg(element: HTMLElement) { + const style = element.style; + const old = parseCSSUrl(style.backgroundImage); + + const target = rewriteURL(old); + if (!target) return; + + const lazyLoader = new Image(); + + lazyLoader.onload = () => { + // Don't swap if a placeholder thumbnail was provided. + // Placeholder thumbnails are the same size as the "default" size. + if (lazyLoader.naturalHeight === 90) return; + + const curr = parseCSSUrl(style.backgroundImage); + + // Don't swap out element image if it has been changed while target image was loading. + if (curr.href !== old.href) return; + + style.backgroundImage = `url(${target.href})`; + }; + + lazyLoader.src = target.href; +} + +const obs = new MutationObserver((mutations) => { + const YT_THUMBNAIL_ELEMENT_TAG = 'ytlr-thumbnail-details'; + + const dummy = document.createElement('div'); + + // handle backgroundImage change + // YT re-uses thumbnail elements in its virtual list implementation. + mutations + .filter((mut) => mut.type === 'attributes') + .map((mut) => [mut.target, mut] as const) + .filter((value): value is [HTMLElement, MutationRecord] => { + const [node, { oldValue }] = value; + dummy.style.cssText = oldValue ?? ''; + + return ( + node instanceof HTMLElement && + node.matches(YT_THUMBNAIL_ELEMENT_TAG) && + node.style.backgroundImage !== '' && + node.style.backgroundImage !== dummy.style.backgroundImage + ); + }) + .map(([elem]) => elem) + .forEach(upgradeBgImg); + + // handle element add + mutations + .filter((mut) => mut.type === 'childList') + .flatMap((mut) => Array.from(mut.addedNodes)) + .filter((node) => node instanceof HTMLElement) + .flatMap((elem) => + Array.from(elem.querySelectorAll(YT_THUMBNAIL_ELEMENT_TAG)) + ) + .filter((elem) => elem.style.backgroundImage !== '') + .forEach(upgradeBgImg); +}); + +function enableObserver() { + obs.observe(document.body, { + subtree: true, + childList: true, + attributes: true, + attributeFilter: ['style'], + attributeOldValue: true + }); +} + +import { configRead, configAddChangeListener } from './config'; + +if (configRead('upgradeThumbnails')) enableObserver(); + +configAddChangeListener('upgradeThumbnails', (value) => + value ? enableObserver() : obs.disconnect() +); diff --git a/src/ui.js b/src/ui.js index aa1bc38..c057099 100644 --- a/src/ui.js +++ b/src/ui.js @@ -124,6 +124,7 @@ function createOptionsPanel() { elmContainer.appendChild(elmHeading); elmContainer.appendChild(createConfigCheckbox('enableAdBlock')); + elmContainer.appendChild(createConfigCheckbox('upgradeThumbnails')); elmContainer.appendChild(createConfigCheckbox('hideLogo')); elmContainer.appendChild(createConfigCheckbox('removeShorts')); elmContainer.appendChild(createConfigCheckbox('enableSponsorBlock')); diff --git a/src/userScript.js b/src/userScript.js index b81fdb0..ec22923 100644 --- a/src/userScript.js +++ b/src/userScript.js @@ -16,7 +16,8 @@ import './adblock.js'; import './shorts.js'; import './sponsorblock.js'; import './ui.js'; -import './font-fix.css' +import './font-fix.css'; +import './thumbnail-quality'; // This IIFE is to keep the video element fill the entire window so that screensaver doesn't kick in. (async () => {