Greasy Fork is available in English.
Hugely improves load speed of pages with lots of embedded Youtube videos by instantly showing clickable and immediately accessible placeholders, then the thumbnails are loaded in background. Optionally a fast simple HTML5 direct playback (720p max) can be selected if available for the video.
// ==UserScript== // @name FYTE /Fast YouTube Embedded/ Player // @description Hugely improves load speed of pages with lots of embedded Youtube videos by instantly showing clickable and immediately accessible placeholders, then the thumbnails are loaded in background. Optionally a fast simple HTML5 direct playback (720p max) can be selected if available for the video. // @description:en Hugely improves load speed of pages with lots of embedded Youtube videos by instantly showing clickable and immediately accessible placeholders, then the thumbnails are loaded in background. Optionally a fast simple HTML5 direct playback (720p max) can be selected if available for the video. // @description:ru На порядок ускоряет время загрузки страниц с большим количеством вставленных Youtube-видео. С первого момента загрузки страницы появляются заглушки для видео, которые можно щелкнуть для загрузки плеера, и почти сразу же появляются кавер-картинки с названием видео. В опциях можно включить режим использования упрощенного браузерного плеера (макс. 720p). // // @version 2.15.1 // // @include * // @exclude /^https:\/\/(www\.)?youtube\.com\/(?!embed)/ // @exclude https://accounts.google.*/o/oauth2/postmessageRelay* // @exclude https://clients*.google.*/youtubei/* // @exclude https://clients*.google.*/static/proxy* // @exclude https://pikabu.ru/* // // @author wOxxOm // @namespace wOxxOm.scripts // @license MIT License // // @grant GM_getValue // @grant GM_listValues // @grant GM_deleteValue // @grant GM_setValue // @grant GM_addStyle // @grant GM_xmlhttpRequest // // @connect www.youtube.com // @connect youtube.com // // @run-at document-start // // @icon  // // @compatible chrome // @compatible firefox // @compatible opera // ==/UserScript== 'use strict'; let localStorage; try { ({localStorage} = window); } catch (e) { localStorage = {}; } // keep video info cache for a month since last time it's shown const CACHE_STALE_DURATION = 30 * 24 * 3600e3; const CACHE_PREFIX = 'FYTE-cache-'; const CACHE_PROPS = [ ['videoWidth', 0], ['videoHeight', 0], ['duration'], ['fps'], ['title'], ['cover'], ]; const CHROME = !CSS.supports('-moz-appearance', 'none'); const rxYoutubeId = /(?:https?:)?\/\/(?:www\.)?(?:youtube(?:-nocookie)?\.com(?=\/)(?:\/embed\/(?:v=)?|.*?[&?/]v[=/])|youtu\.be\/)([-\w]+)[^'"\s]*/; const rxYoutubeIdHtml = new RegExp(`${/(?:src|value)\s*=\s*["']\s*/.source}(${rxYoutubeId.source})['"]|$`, 'i'); const cfg = { width: 1280, height: 720, invidious: false, resize: 'Fit to width', rules: {}, pinnable: 'on', pinnedWidth: 400, playHTML5: false, playHTML5Shown: false, showStoryboard: true, skipCustom: true, }; const checked = new WeakMap(); const fyteMap = new WeakMap(); const dbCache = {}; const dbFlush = new Set(); const dbFlushDelay = 1000; let _, db, fytedom, styledom, iframes, objects, persite, playbtn; if (location.hostname === 'www.youtube.com') { if (CHROME && window !== top) setupYoutubeFullscreenRelay(); } else { for (const [k, def] of Object.entries(cfg)) { const v = GM_getValue(k, def); cfg[k] = typeof v === typeof def ? v : def; } _ = initTL(); persite = getPersiteRule(); fytedom = document.getElementsByClassName('instant-youtube-container'); iframes = document.getElementsByTagName('iframe'); objects = document.getElementsByTagName('object'); updateCustomSize(); findEmbeds([]); injectStylesIfNeeded(); new MutationObserver(findEmbeds) .observe(document, {subtree: true, childList: true}); document.addEventListener('DOMContentLoaded', e => { injectStylesIfNeeded(); adjustNodesIfNeeded(e); }, {once: true}); addEventListener('resize', adjustNodesIfNeeded, true); addEventListener('message', onMessageHost); } function setupYoutubeFullscreenRelay() { parent.postMessage('FYTE-toggle-fullscreen-init', '*'); addEventListener('message', function onMessage(e) { if (e.source !== parent || e.data !== 'FYTE-toggle-fullscreen-init-confirmed') return; removeEventListener('message', onMessage); const fsbtn = document.getElementsByClassName('ytp-fullscreen-button'); new MutationObserver(function () { const el = fsbtn[0]; if (el) { this.disconnect(); el.removeAttribute('aria-disabled'); el.replaceWith(el.cloneNode(true)); fsbtn[0].addEventListener('click', () => parent.postMessage('FYTE-toggle-fullscreen', '*')); } }).observe(document, {subtree: true, childList: true}); }); } function getPersiteRule() { const h = location.hostname; const rule = (cfg.rules || {})[h] || h === 'developers.google.com' && { test: '[data-video-id]', src: e => '//youtu.be/' + e.dataset.videoId, } || h === 'play.google.com' && { eatparent: 0, } || h === 'androidauthority.com' && { eatparent: '.video-container', } || h === 'reddit.com' && { test: '[data-url*="youtube.com/"], [data-url*="youtu.be/"]', has: '[src*="/mediaembed"]', attr: 'data-url', } || h === '9gag.com' && { eatparent: 0, } || h === 'anilist.co' && { eatparent: '.youtube', } || h === 'www.theverge.com' && { eatparent: '.p-scalable-video', } || /^www\.google\.\w{2,3}(\.\w{2,3})?$/.test(h) && $('html[itemtype$="SearchR###ltsPage"]') && { find: '#rcnt a[data-attrid="VisualDigestVideoR###lt"][href*="youtube.com/watch"]', test: '', eatparent: 2, }; if (!rule) return; Object.setPrototypeOf(rule, null); const {find = 'iframe', has, test: q = '[src*="youtube.com/embed"]'} = rule; if (!/^\.?[a-z][-a-z]*$/i.test(find)) Object.defineProperty(rule, 'find', {get: () => document.querySelectorAll(find)}); else rule.find = find[0] === '.' ? document.getElementsByClassName(find.slice(1)) : document.getElementsByTagName(find); if (q) rule.test = el => (el = el.matches(q) ? el : el.querySelector(q)) && (!has || el.querySelector(has)) && el; return rule; } function onMessageHost(e) { switch (e.data) { case 'FYTE-toggle-fullscreen-init': if (findFrameElement(e.source)) e.source.postMessage('FYTE-toggle-fullscreen-init-confirmed', '*'); break; case 'FYTE-toggle-fullscreen': { const el = findFrameElement(e.source); if (el) goFullscreen(el, !(document.fullscreenElement || document.fullScreen || document.mozFullScreen)); break; } case 'iframe-allowfs': $$('iframe:not([allowfullscreen])').some(iframe => { if (iframe.contentWindow === e.source) { iframe.allowFullscreen = true; return true; } }); if (window !== top) parent.postMessage('iframe-allowfs', '*'); break; } } function findFrameElement(frameWindow) { return $$('iframe[allowfullscreen]').find(el => el.contentWindow === frameWindow); } async function findEmbeds(mutations) { const found = []; if (mutations.length === 1) { const added = mutations[0].addedNodes; if (!added[0] || !added[1] && added[0].nodeType === 3) return; } if (persite) for (let el of persite.find) if (!persite.test || (el = persite.test(el))) processEmbed(found, el, persite.src ? persite.src(el) : el.getAttribute(persite.attr)); if (length && hasChildWindow()) { for (const el of iframes) { const src = rxYoutubeId.exec(el.dataset.src || el.src); if (src) processEmbed(found, el, src[0]); } for (const el of objects) { const src = el.innerHTML.match(rxYoutubeIdHtml)[1]; if (src) processEmbed(found, el, src); } } if (!found.length) return; const toRead = []; for (const [id] of found) if (!dbCache[id] && !toRead.includes(id)) toRead.push(id); if (toRead.length) await read(toRead); for (const r of found) createFYTE(...r); } function hasChildWindow() { for (let i = 0, w; i < length; i++) if ((w = unsafeWindow[i]) && typeof w === 'object' && !checked.has(w)) return checked.set(w, 0); } function decodeEmbedUrl(url) { return /youtube(-nocookie)?\.com%2Fembed/.test(url) ? decodeURIComponent(url.replace(/^.*?(http[^&?=]+?youtube(-nocookie)?\.com%2Fembed[^&]+).*$/i, '$1')) : url; } function processEmbed(res, node, src) { src = src || node.src || node.href || ''; if (!src || checked.get(node) === src) return; checked.set(node, src); let n = node; let np = n.parentNode; const srcFixed = decodeEmbedUrl(src) .replace(/\/(watch\?v=|v\/)/, '/embed/') .replace(/^([^?&]+)&/, '$1?'); if (src.indexOf('cdn.embedly.com/') > 0 || cfg.resize !== 'Original' && np && np.children.length === 1 && !np.className && !np.id) { n = location.hostname === 'disqus.com' ? np.parentNode : np; np = n.parentElement; } if (!np || !np.parentNode || cfg.skipCustom && srcFixed.includes('enablejsapi=1') || srcFixed.includes('/embed/videoseries') || node.matches('.instant-youtube-embed, .YTLT-embed, .ihvyoutube') || node.style.position === 'fixed' || node.onload // skip some retarded loaders ) return; let id = srcFixed.match(rxYoutubeId); if (!id) return; id = id[1]; if (np.localName === 'object') { n = np; np = n.parentElement; } let eatparent = persite && persite.eatparent || 0; if (typeof eatparent === 'string') { n = np.closest(eatparent) || n; } else { while (eatparent--) { n = np; np = n.parentElement; } } (n.contentWindow || {}).location = 'about:blank'; res.push([id, n, srcFixed]); } function createFYTE(id, n, srcFixed) { const cache = dbCache[id] || {id}; const autoplay = /[?&](autoplay=1|ps=play)(&|$)/.test(srcFixed); const div = $create('div.container'); const img = $create('img.thumbnail', {style: important('opacity:0;')}); if (!autoplay) { img.src = cache.cover || getCoverUrl(id, 'maxresdefault.jpg'); img.onload = onCoverLoad; img.onerror = onCoverError; } injectStylesIfNeeded('force'); const fyte = { state: 'querying', srcEmbed: srcFixed.replace(/&$/, ''), originalWidth: /%/.test(n.width) ? 320 : n.width | 0 || n.clientWidth | 0, originalHeight: /%/.test(n.height) ? 200 : n.height | 0 || n.clientHeight | 0, cache, }; fyteMap.set(div, fyte); fyte.srcEmbedFixed = fyte.srcEmbed.replace(/^http:/, 'https:') .replace(/([&?])(wmode=\w+|feature=oembed)&?/, '$1') .replace(/[&?]$/, ''); fyte.srcWatchFixed = fyte.srcEmbedFixed.replace('/embed/', '/watch?v=').replace(/(\?.*?)\?/, '$1&'); cache.lastUsed = new Date(); write(cache); if (cache.reason) div.setAttribute('disabled', ''); const divSize = calcContainerSize(div, n, fyte); const origStyle = getComputedStyle(n); overrideCSS(div, Object.assign( { height: persite && persite.eatparent === 0 ? '100%' : divSize.h + 'px', 'min-width': Math.min(divSize.w, fyte.originalWidth) + 'px', 'min-height': Math.min(divSize.h, fyte.originalHeight) + 'px', 'max-width': divSize.w + 'px', }, origStyle.transform && { transform: origStyle.transform, }, !autoplay && { 'background-color': 'transparent', transition: 'background-color 2s', }, // eslint-disable-next-line no-proto ...Object.keys(origStyle.hasOwnProperty('position') ? origStyle : origStyle.__proto__ /*FF*/) .filter(k => /^(position|left|right|top|bottom)$/.test(k) && !/^(auto|static|block)$/.test(origStyle[k])) .map(k => ({[k]: origStyle[k]})), origStyle.display === 'inline' && { display: 'inline-block', width: '100%', }, cfg.resize === 'Fit to width' && { width: '100%', })); if (!autoplay) { setTimeout(() => div.style.removeProperty('background-color')); setTimeout(() => div.style.removeProperty('transition'), 2000); } const wrapper = $create('div.wrapper', {}, [ img, $create('a.title', {target: '_blank', href: fyte.srcWatchFixed}, cache.title || cache.reason ? [ $create('strong', {}, cache.title || cache.reason || ''), cache.duration && $create('span', {}, cache.duration), cache.fps && $create('i', {}, `${cache.fps}fps`), ] : '\xA0'), (playbtn || initPlayButton()).cloneNode(true), $create('span.alternative', {}, _(`msgPlay${cfg.playHTML5 ? 'HTML5' : ''}`)), $create('div.storyboard', {hidden: !cfg.showStoryboard}), $create('div.options-button', {}, _('Options')), ]); div.appendChild(wrapper); fyteMap.set(img, autoplay); if (!autoplay && img.naturalWidth) img.onload(); n.replaceWith(div); if (!cache.title && !cache.reason || autoplay && cfg.playHTML5) fetchInfo.call(div); if (autoplay) { startPlaying(div); } else { div.addEventListener('click', clickHandler); div.addEventListener('mousedown', clickHandler); div.addEventListener('mouseenter', fetchInfo); } if (cfg.showStoryboard) div.addEventListener('mousemove', trackMouse); } function fetchInfo(e) { const fyte = fyteMap.get(this); fyte.mouseEvent = e; this.removeEventListener('mouseenter', fetchInfo); if (!fyte.storyboard) { const {id} = fyte.cache; GM_xmlhttpRequest({ method: 'GET', url: 'https://www.youtube.com/watch?v=' + id, context: this, onload: parseVideoInfo, }); } } function onCoverLoad(e) { if (this.naturalWidth <= 120) return this.onerror(e); this.style.opacity = ''; fyteMap.delete(this); } function onCoverError() { const {src} = this; const id = src.split('/')[4]; const src2 = getCoverUrl(id, 'sddefault.jpg'); this.src = src2 !== src ? src2 : getCoverUrl(id, 'hqdefault.jpg'); } function adjustNodesIfNeeded(e) { if (!fytedom[0]) return; if (adjustNodesIfNeeded.scheduled) clearTimeout(adjustNodesIfNeeded.scheduled); adjustNodesIfNeeded.scheduled = setTimeout(() => { adjustNodes(e); adjustNodesIfNeeded.scheduled = 0; }, 16); } function adjustNodes(event, clickedContainer) { const force = !!clickedContainer; let nearest = force ? clickedContainer : null; let nearestCenterYpct; const vids = $$('.instant-youtube-container:not([pinned]):not([stub])'); if (!nearest && event.type !== 'DOMContentLoaded') { let minDistance = window.innerHeight * 3 / 4 | 0; const nearTargetY = window.innerHeight / 2; for (const n of vids) { const bounds = n.getBoundingClientRect(); const distance = Math.abs((bounds.bottom + bounds.top) / 2 - nearTargetY); if (distance < minDistance) { minDistance = distance; nearest = n; } } } if (nearest) { const bounds = nearest.getBoundingClientRect(); nearestCenterYpct = (bounds.top + bounds.bottom) / 2 / window.innerHeight; } let resized = false; for (const n of vids) { const size = calcContainerSize(n); const w = size.w; const h = size.h; // prevent parent clipping for (let e = n.parentElement, style; e; e = e.parentElement) { if (e.style.overflow !== 'visible' && n.offsetTop < e.clientHeight / 2 && n.offsetTop + n.clientHeight > e.clientHeight && (style = getComputedStyle(e)) && /hidden|scroll/.test(style.overflow + style.overflowX + style.overflowY)) { overrideCSS(e, { overflow: 'visible', 'overflow-x': 'visible', 'overflow-y': 'visible', }); } } if (force && Math.abs(w - parseFloat(n.style.maxWidth)) <= 2) continue; overrideCSS(n, Object.assign({}, n.style.maxWidth !== `${w}px` && { 'max-width': `${w}px`, }, n.style.height !== h + 'px' && { height: h + 'px', }, parseFloat(n.style.minWidth) > w && { 'min-width': n.style.maxWidth, }, parseFloat(n.style.minHeight) > h && { 'min-height': n.style.height, })); resized = true; } if (resized && nearest) setTimeout(() => { const bounds = nearest.getBoundingClientRect(); const h = bounds.bottom - bounds.top; const projectedCenterY = nearestCenterYpct * window.innerHeight; const projectedTop = projectedCenterY - h / 2; const safeTop = Math.min(Math.max(0, projectedTop), window.innerHeight - h); window.scrollBy(0, bounds.top - safeTop); }, 16); } function calcContainerSize(div, origNode, fyte = fyteMap.get(div)) { if (!fyte) return; origNode = origNode || div; let w, h; let np = origNode.parentElement; const style = getComputedStyle(np); let parentWidth = parseFloat(style.width) - floatPadding(np, style, 'Left') - floatPadding(np, style, 'Right'); if (+style.columnCount > 1) parentWidth = (parentWidth + parseFloat(style.columnGap)) / style.columnCount - parseFloat(style.columnGap); switch (cfg.resize) { case 'Original': if (fyte.originalWidth === 320 && fyte.originalHeight === 200) { w = parentWidth; h = parentWidth / 16 * 9; } else { w = fyte.originalWidth; h = fyte.originalHeight; } break; case 'Custom': w = cfg.width; h = cfg.height; break; case '1080p': case '720p': case '480p': case '360p': h = parseInt(cfg.resize); w = h / 9 * 16; break; default: // fit-to-width mode // find parent node with nonzero width (i.e. independent of our video element) while (np && !(w = np.clientWidth)) np = np.parentNode; if (w) h = w / 16 * 9; else { w = origNode.clientWidth; h = origNode.clientHeight; } } if (parentWidth > 0 && parentWidth < w) { h *= parentWidth / w; w = parentWidth; } if (cfg.resize === 'Fit to width' && h < fyte.originalHeight * 0.9) h = Math.min(fyte.originalHeight, w / fyte.originalWidth * fyte.originalHeight); return {w: CHROME ? w : Math.round(w), h: h}; } function parseVideoInfo(response) { const div = response.context; const txt = response.responseText; const info = tryJSONparse(txt.match(/var\s+ytInitialPlayerResponse\s*=\s*({.+?});|$/)[1]) || {}; const {reason} = info.playabilityStatus || {}; const vid = info.videoDetails || {}; const streams = info.streamingData || {}; const fyte = fyteMap.get(div); const cache = fyte.cache; let shouldUpdateCache = false; const videoSources = []; const fmts = (streams.formats || streams.adaptiveFormats || []) .sort((a, b) => b.width - a.width || b.height - a.height); // parse width & height to adjust the thumbnail if (fmts.length && (cache.videoWidth !== fmts[0].width || cache.videoHeight !== fmts[0].height)) { cache.videoWidth = fmts[0].width; cache.videoHeight = fmts[0].height; shouldUpdateCache = true; } // parse video sources for (const f of fmts) { const codec = f.mimeType.match(/codecs="([^.]+)|$/)[1] || ''; const type = f.mimeType.split(/[/;]/)[1]; let src = f.url; if (!src && f.cipher) { const sp = {}; for (const str of f.cipher.split('&')) { const [k, v] = str.split('='); sp[k] = v; } src = decodeURIComponent(sp.url); if (sp.s) src += `&${sp.sp || 'sig'}=${decodeYoutubeSignature(sp.s)}`; } videoSources.push({ src, title: [ f.quality, f.qualityLabel !== f.quality ? f.qualityLabel : '', type + (codec ? `:${codec}` : ''), ].filter(Boolean).join(', '), }); } let fps = new Set(); for (const f of streams.adaptiveFormats || []) { if (f.fps) fps.add(f.fps); } fps = [...fps].join('/'); if (fps && cache.fps !== fps) { cache.fps = fps; shouldUpdateCache = true; } let duration = fyte.duration = vid.lengthSeconds | 0; if (duration) { duration = secondsToTimeString(duration); if (cache.duration !== duration) { cache.duration = duration; shouldUpdateCache = true; } } if (duration || fps) duration = `<span>${duration}</span>${fps ? `<i>${fps}fps</i>` : ''}`; const title = [ decodeURIComponent(vid.title || ''), reason && reason.replace(/\s*\.$/, ''), ].filter(Boolean).join(' | ').replace(/\+/g, ' '); if (title) { $('.instant-youtube-title', div).innerHTML = (title ? `<strong>${title}</strong>` : '') + duration; if (cache.title !== title) { cache.title = title; shouldUpdateCache = true; } } if (cfg.pinnable !== 'off' && vid.title) makeDraggable(div); if (reason) { div.setAttribute('disabled', ''); if (cache.reason !== reason) { cache.reason = reason; shouldUpdateCache = true; } } if (videoSources.length) fyte.videoSources = videoSources; if (txt.includes('playerStoryboardSpecRenderer') && info.storyboards && fyte.state !== 'scheduled play') { const m = info.storyboards.playerStoryboardSpecRenderer.spec.split('|'); const [w, h, len, rows, cols] = m[m.length - 1].split('#').map(Number); fyte.storyboard = {w, h, len, rows, cols}; if (w * h > 2000) { fyte.storyboard.url = m[0].replace('?', '&').replace( '$L/$N.jpg', `${m.length - 2}/M0.jpg?sigh=${m[m.length - 1].replace(/^.+?#([^#]+)$/, '$1')}`); const elSb = $('.instant-youtube-storyboard', div); if (elSb) { elSb.dataset.loaded = ''; elSb.appendChild(overrideCSS($create('div.sb-thumb', {}, '\xA0'), { width: w - 1 + 'px', height: h + 'px', })); if (cfg.showStoryboard) updateHoverHandler(div); } } } injectStylesIfNeeded(); if (fyte.state === 'scheduled play') setTimeout(startPlayingDirectly, 0, div); fyte.state = ''; try { const cover = vid.thumbnail.thumbnails.pop().url; if (cache.cover !== cover) { cache.cover = cover; shouldUpdateCache = true; const img = $('img', div); if (img.src && img.src !== cover) img.src = cover; } } catch (e) {} if (shouldUpdateCache) write(cache); } function decodeYoutubeSignature(s) { const a = s.split(''); a.reverse(); swap(a, 24); a.reverse(); swap(a, 41); a.reverse(); swap(a, 2); return a.join(''); } function swap(a, b) { const c = a[0]; a[0] = a[b % a.length]; a[b % a.length] = c; } function trackMouse(e) { fyteMap.get(this).mouseEvent = e; } function updateHoverHandler(div) { const fyte = fyteMap.get(div); const sb = fyte.storyboard; const elSb = $('.instant-youtube-storyboard', div); if (!cfg.showStoryboard) { elSb.hidden = true; return; } elSb.hidden = false; let oldIndex = null; const tracker = elSb.firstElementChild; const style = tracker.style; const sbImg = $create('img'); const spinner = $create('span.loading-spinner'); elSb.addEventListener('mousemove', storyboardHoverHandler); elSb.addEventListener('mouseout', storyboardHoverHandler); elSb.addEventListener('click', storyboardClickHandler, {once: true}); div.addEventListener('mouseover', storyboardPreloader); div.addEventListener('mouseout', storyboardPreloader); if (div.closest(':hover')) storyboardPreloader({}); function storyboardClickHandler(e) { const offsetX = e.offsetX || e.clientX - elSb.getBoundingClientRect().left; fyte.startAt = offsetX / elSb.clientWidth * fyte.duration | 0; fyte.srcEmbedFixed = setUrlParams(fyte.srcEmbedFixed, {start: fyte.startAt}); startPlaying(div, {alternateMode: e.shiftKey}); } function storyboardPreloader(e) { if (e.type === 'mouseout') { spinner.remove(); return; } const {len, rows, cols, preloaded} = sb || {}; const lastpart = (len - 1) / (rows * cols || 1) | 0; if (lastpart <= 0 || preloaded) return; let part = 0; $create('img', { src: setStoryboardUrl(part++), onload() { if (part <= lastpart) { this.src = setStoryboardUrl(part++); return; } sb.preloaded = true; div.removeEventListener('mouseover', storyboardPreloader); div.removeEventListener('mouseout', storyboardPreloader); this.onload = null; this.src = ''; spinner.remove(); }, }); if (elSb.matches(':hover') && fyte.mouseEvent) storyboardHoverHandler(fyte.mouseEvent); } function setStoryboardUrl(part) { return sb.url.replace(/M\d+\.jpg\?/, `M${part}.jpg?`); } function storyboardHoverHandler(e) { div.removeEventListener('mousemove', trackMouse); if (!cfg.showStoryboard || !sb) return; if (e.type === 'mouseout') { sbImg.onload && sbImg.onload(); return; } const {w, h, cols, rows, len, preloaded} = sb; const partlen = rows * cols; const offsetX = e.offsetX || e.clientX - elSb.getBoundingClientRect().left; const left = Math.min(elSb.clientWidth - w, Math.max(0, offsetX - w)) | 0; if (!style.left || parseInt(style.left) !== left) { style.left = `${left}px`; if (spinner.parentElement) spinner.style.cssText = important(`left:${left + w / 2 - 10}px; right:auto;`); } let index = Math.min(offsetX / elSb.clientWidth * (len + 1) | 0, len - 1); if (index === oldIndex) return; const part = index / partlen | 0; if (!oldIndex || part !== (oldIndex / partlen | 0)) { const url = setStoryboardUrl(part); style.setProperty('background-image', `url(${url})`, 'important'); if (!preloaded) { if (spinner.timer) clearTimeout(spinner.timer); spinner.timer = setTimeout(() => { spinner.timer = 0; if (!sbImg.src) return; elSb.appendChild(spinner); spinner.style.cssText = important(`left:${left + w / 2 - 10}px; right:auto;`); }, 50); sbImg.onload = () => { clearTimeout(spinner.timer); spinner.remove(); spinner.timer = 0; sbImg.onload = null; sbImg.src = ''; }; sbImg.src = url; } } tracker.dataset.time = secondsToTimeString(index / (len - 1 || 1) * fyte.duration | 0); oldIndex = index; index %= partlen; style.setProperty('background-position', `-${(index % cols) * w}px -${(index / cols | 0) * h}px`, 'important'); } } function clickHandler(e) { const el = e.target; if (el.closest('a') || e.type === 'mousedown' && e.button !== 1 || e.type === 'click' && el.matches('.instant-youtube-options, .instant-youtube-options *')) return; if (e.type === 'click' && el.matches('.instant-youtube-options-button')) { showOptions(e); e.preventDefault(); e.stopPropagation(); return; } e.preventDefault(); e.stopPropagation(); e.stopImmediatePropagation(); startPlaying(el.closest('.instant-youtube-container'), { alternateMode: e.shiftKey || el.matches('.instant-youtube-alternative'), fullscreen: e.button === 1, }); } function startPlaying(div, params) { div.removeEventListener('click', clickHandler); div.removeEventListener('mousedown', clickHandler); $$remove([ '.instant-youtube-alternative', '.instant-youtube-storyboard', '.instant-youtube-options-button', '.instant-youtube-options', ].join(','), div); $('svg', div).outerHTML = '<span class=instant-youtube-loading-spinner></span>'; if (cfg.pinnable !== 'off') { makePinnable(div); if (params && params.pin) $(`[pin="${params.pin}"]`, div).click(); } if (window !== top) parent.postMessage('iframe-allowfs', '*'); const fyte = fyteMap.get(div); if ((!!cfg.playHTML5 + !!(params && params.alternateMode) === 1) && (fyte.videoSources || fyte.state === 'querying')) { if (fyte.videoSources) startPlayingDirectly(div, params); else { // playback will start in parseVideoInfo fyte.state = 'scheduled play'; // fallback to iframe in 5s setTimeout(() => { if (fyte.state) { fyte.state = ''; switchToIFrame.call(div, params); } }, 5000); } } else switchToIFrame.call(div, params); } function startPlayingDirectly(div, params) { const switchTimer = setTimeout(switchToIFrame.bind(div, params), 5000); const video = $create('video.embed', { autoplay: true, controls: true, volume: GM_getValue('volume', 0.5), style: { position: 'absolute', left: 0, top: 0, right: 0, bottom: 0, padding: 0, margin: 'auto', opacity: 0, width: '100%', height: '100%', }, oncanplay() { this.oncanplay = null; const fyte = fyteMap.get(div); if (fyte.startAt && Math.abs(this.currentTime - fyte.startAt) > 1) this.currentTime = fyte.startAt; clearTimeout(switchTimer); pauseOtherVideos(this); if (params && params.fullscreen) return; div.setAttribute('playing', ''); div.firstElementChild.appendChild(this); overrideCSS(this, {opacity: 1}); }, onvolumechange() { GM_setValue('volume', this.volume); }, }); for (const src of fyteMap.get(div).videoSources || []) { video.appendChild($create('source', src)) .onerror = switchToIFrame.bind(div, params); } overrideCSS($('img', div), { transition: 'opacity 1s', opacity: '0', }); if (params && params.fullscreen) { div.firstElementChild.appendChild(video); div.setAttribute('playing', ''); video.style.opacity = 1; goFullscreen(video); } if (CHROME && +navigator.userAgent.match(/Chrom\D+(\d+)|$/)[1] < 74) video.addEventListener('click', () => setTimeout(() => video.paused ? video.play() : video.pause())); const title = $('.instant-youtube-title', div); if (title) { video.onpause = () => (title.hidden = false); video.onplay = () => (title.hidden = true); } } function switchToIFrame(params, e) { if (this.querySelector('iframe')) return; const div = this; const wrapper = div.firstElementChild; const fullscreen = params && params.fullscreen && !e; const fyte = fyteMap.get(div); if (e instanceof Event) { console.log('[FYTE] Direct linking canceled on %s, switching to IFRAME player', fyte.srcEmbed); const video = e.target ? e.target.closest('video') : e.composedPath().pop(); video.textContent = ''; goFullscreen(video, false); video.remove(); } const url = setUrlParams(fyte.srcEmbedFixed, { html5: 1, autoplay: 1, autohide: 2, border: 0, controls: 1, fs: 1, showinfo: 1, ssl: 1, theme: 'dark', enablejsapi: 1, local: 'true', quality: 'medium', FYTEfullscreen: fullscreen | 0, }); let iframe = $create('iframe.embed', { src: url, allow: 'autoplay; fullscreen', allowFullscreen: true, width: '100%', height: '100%', style: { position: 'absolute', top: 0, left: 0, right: 0, padding: 0, margin: 'auto', opacity: 0, border: 0, }, }); if (cfg.pinnable !== 'off') { $('[pin]', div).insertAdjacentElement('beforebegin', iframe); } else { wrapper.appendChild(iframe); } div.setAttribute('iframe', ''); div.setAttribute('playing', ''); iframe = $('iframe', div); if (fullscreen) { goFullscreen(iframe); overrideCSS(iframe, {opacity: 1}); } addEventListener('message', YTlistener); iframe.addEventListener('load', () => { iframe.contentWindow.postMessage('{"event":"listening"}', '*'); if (fyte.cache.reason) show(); }, {once: true}); setTimeout(show, 1000); function show() { overrideCSS(iframe, {opacity: 1}); $('.instant-youtube-title', div).hidden = true; } function YTlistener(e) { const data = e.source === iframe.contentWindow && e.data && tryJSONparse(e.data); if (!data || !data.info || data.info.playerState !== 1) return; removeEventListener('message', YTlistener); pauseOtherVideos(iframe); overrideCSS(iframe, {opacity: 1}); overrideCSS($('img', div), {display: 'none'}); $$remove('span, a', div); } } function getCoverUrl(id, name) { return `https://i.ytimg.com/vi${name.endsWith('.webp') ? '_webp' : ''}/${id}/${name}`; } function setUrlParams(url, params) { const u = new URL(url); for (const [k, v] of Object.entries(params)) u.searchParams.set(k, v); return u.href; } function pauseOtherVideos(activePlayer) { for (const v of $$('.instant-youtube-embed', activePlayer.ownerDocument)) { if (v === activePlayer) continue; switch (v.localName) { case 'video': if (!v.paused) v.pause(); break; case 'iframe': try { v.contentWindow.postMessage('{"event":"command", "func":"pauseVideo", "args":""}', '*'); } catch (e) {} break; } } } function goFullscreen(el, enable) { if (enable !== false) el.webkitRequestFullScreen && el.webkitRequestFullScreen() || el.mozRequestFullScreen && el.mozRequestFullScreen() || el.requestFullScreen && el.requestFullScreen(); else document.webkitCancelFullScreen && document.webkitCancelFullScreen() || document.mozCancelFullScreen && document.mozCancelFullScreen() || document.cancelFullScreen && document.cancelFullScreen(); } function makePinnable(div) { div.firstElementChild.insertAdjacentHTML('beforeend', '<div size-gripper></div>' + '<div pin="top-left"></div>' + '<div pin="top-right"></div>' + '<div pin="bottom-right"></div>' + '<div pin="bottom-left"></div>'); for (const pin of $$('[pin]', div)) { if (cfg.pinnable === 'hide') pin.setAttribute('transparent', ''); pin.onclick = pinClicked; } $('[size-gripper]', div).addEventListener('mousedown', startResize, true); function pinClicked() { const pin = this; const pinIt = !div.hasAttribute('pinned') || !pin.hasAttribute('active'); const corner = pin.getAttribute('pin'); const video = $('video', div); const paused = video.paused; const fyte = fyteMap.get(div); if (pinIt) { for (const p of $$('[pin][active]', div)) p.removeAttribute('active'); pin.setAttribute('active', ''); if (!fyte.unpinnedStyle) { fyte.unpinnedStyle = div.style.cssText; const stub = div.cloneNode(); const img = $('img', div).cloneNode(); img.style.opacity = 1; img.style.display = 'block'; img.title = ''; stub.appendChild(img); stub.onclick = e => $('[pin][active]', div).onclick(e); stub.style.setProperty('opacity', .3, 'important'); stub.setAttribute('stub', ''); fyte.stub = stub; div.parentNode.insertBefore(stub, div); } const size = constrainPinnedSize(div, localStorage.FYTEwidth || cfg.pinnedWidth); overrideCSS(div, { position: 'fixed', width: size.w + 'px', height: size.h + 'px', top: corner.includes('top') ? 0 : 'auto', left: corner.includes('left') ? 0 : 'auto', right: corner.includes('right') ? 0 : 'auto', bottom: corner.includes('bottom') ? 0 : 'auto', 'z-index': 999999999, }); adjustPinnedOffset(div, div, corner); div.setAttribute('pinned', corner); if (video && document.body) document.body.appendChild(div); } else { // unpin pin.removeAttribute('active'); div.removeAttribute('pinned'); div.style.cssText = fyte.unpinnedStyle; fyte.unpinnedStyle = ''; if (fyte.stub) { if (video && document.body) fyte.stub.parentNode.replaceChild(div, fyte.stub); fyte.stub.remove(); fyte.stub = null; } } if (paused) video.pause(); } function startResize(e) { const siteSaved = localStorage.FYTEwidth; let saveAs = siteSaved ? 'site' : 'global'; const oldSizeCSS = {w: div.style.width, h: div.style.height}; const oldDraggable = div.draggable; div.draggable = false; const gripper = this; gripper.removeAttribute('tried-exceeding'); gripper.innerHTML = `<div> <div save-as="${saveAs}"><b>S</b> = Site mode: <span>${getSiteOnlyText()}</span></div> ${!siteSaved ? '' : '<div><b>R</b> = Reset to global size</div>'} <div><b>Esc</b> = Cancel</div> </div>`; document.addEventListener('mousemove', resize); document.addEventListener('mouseup', resizeDone); document.addEventListener('keydown', resizeKeyDown); e.stopImmediatePropagation(); return false; function getSiteOnlyText() { return saveAs === 'site' ? `only ${location.hostname}` : 'global'; } function resize(e) { let deltaX = e.movementX || e.webkitMovementX || e.mozMovementX || 0; if (/right/.test(div.getAttribute('pinned'))) deltaX = -deltaX; const newSize = constrainPinnedSize(div, div.clientWidth + deltaX); if (newSize.w !== div.clientWidth) { div.style.setProperty('width', newSize.w + 'px', 'important'); div.style.setProperty('height', newSize.h + 'px', 'important'); gripper.removeAttribute('tried-exceeding'); } else if (newSize.triedExceeding) { gripper.setAttribute('tried-exceeding', ''); } window.getSelection().removeAllRanges(); return false; } function resizeDone() { div.draggable = oldDraggable; gripper.removeAttribute('tried-exceeding'); gripper.innerHTML = ''; document.removeEventListener('mousemove', resize); document.removeEventListener('mouseup', resizeDone); document.removeEventListener('keydown', resizeKeyDown); switch (saveAs) { case 'site': localStorage.FYTEwidth = div.clientWidth; break; case 'global': cfg.pinnedWidth = div.clientWidth; GM_setValue('pinnedWidth', cfg.pinnedWidth); // fallthrough to remove the locally saved value case 'reset': delete localStorage.FYTEwidth; break; case '': return false; } gripper.setAttribute('saveAs', saveAs); setTimeout(() => gripper.removeAttribute('saveAs'), 250); return false; } function resizeKeyDown(e) { switch (e.code) { case 'Escape': saveAs = 'cancel'; div.style.width = oldSizeCSS.w; div.style.height = oldSizeCSS.h; break; case 'KeyS': saveAs = saveAs === 'site' ? 'global' : 'site'; $('[save-as]', gripper).setAttribute('save-as', saveAs); $('[save-as] span', gripper).textContent = getSiteOnlyText(); return false; case 'KeyR': { if (!siteSaved) return; saveAs = 'reset'; const {w, h} = constrainPinnedSize(div, cfg.pinnedWidth); div.style.width = w; div.style.height = h; break; } default: return; } document.dispatchEvent(new MouseEvent('mouseup')); return false; } } } function makeDraggable(div) { div.draggable = true; div.addEventListener('dragstart', e => { const offsetY = e.offsetY || e.clientY - div.getBoundingClientRect().top; if (offsetY > div.clientHeight - 30) { e.preventDefault(); return; } e.dataTransfer.setData('text/plain', ''); let dropZone = $create('div.dragndrop-placeholder'); const fyte = fyteMap.get(div); const dropZoneHeight = 400 / fyte.cache.videoWidth * fyte.cache.videoHeight; document.body.addEventListener('dragenter', dragHandler); document.body.addEventListener('dragover', dragHandler); document.body.addEventListener('dragend', dragHandler); document.body.addEventListener('drop', dragHandler); function dragHandler(e) { e.stopImmediatePropagation(); e.stopPropagation(); e.preventDefault(); switch (e.type) { case 'dragover': { const playing = div.hasAttribute('playing'); const stub = e.target.closest('.instant-youtube-container[stub]') === fyte.stub && fyte.stub; const gizmo = playing && !stub ? {left: 0, top: 0, right: innerWidth, bottom: innerHeight} : (stub || div).getBoundingClientRect(); const x = e.clientX; const y = e.clientY; const cx = (gizmo.left + gizmo.right) / 2; const cy = (gizmo.top + gizmo.bottom) / 2; const stay = !!stub || y >= cy - 200 && y <= cy + 200 && x >= cx - 200 && x <= cx + 200; overrideCSS(dropZone, { top: y < cy || stay ? '0' : 'auto', bottom: y > cy || stay ? '0' : 'auto', left: x < cx || stay ? '0' : 'auto', right: x > cx || stay ? '0' : 'auto', width: playing && stay && stub ? stub.clientWidth + 'px' : '400px', height: playing && stay && stub ? stub.clientHeight + 'px' : dropZoneHeight + 'px', margin: playing && stay ? 'auto' : '0', position: !playing && stay || stub ? 'absolute' : 'fixed', 'background-color': stub ? 'rgba(0,0,255,0.5)' : stay ? 'rgba(255,255,0,0.4)' : 'rgba(0,255,0,0.2)', }); adjustPinnedOffset(dropZone, div); (stay && !playing || stub ? (stub || div) : document.body).appendChild(dropZone); break; } case 'dragend': case 'drop': { const corner = calcPinnedCorner(dropZone); dropZone.remove(); dropZone = null; document.body.removeEventListener('dragenter', dragHandler); document.body.removeEventListener('dragover', dragHandler); document.body.removeEventListener('dragend', dragHandler); document.body.removeEventListener('drop', dragHandler); if (e.type === 'dragend') break; if (div.hasAttribute('playing')) (corner ? $(`[pin="${corner}"]`, div) : fyte.stub).click(); else startPlaying(div, {pin: corner}); } } } }); } function adjustPinnedOffset(el, self, corner) { let offset = 0; if (!corner) corner = calcPinnedCorner(el); for (const pin of $$(`.instant-youtube-container[pinned] [pin="${corner}"][active]`)) { const container = pin.closest('[pinned]'); if (container !== el && container !== self) { const {top, bottom} = container.getBoundingClientRect(); offset = Math.max(offset, el.style.top === '0px' ? bottom : innerHeight - top); } } if (offset) el.style[el.style.top === '0px' ? 'top' : 'bottom'] = offset + 'px'; } function calcPinnedCorner(el) { const t = el.style.top !== 'auto'; const l = el.style.left !== 'auto'; const r = el.style.right !== 'auto'; const b = el.style.bottom !== 'auto'; return t && b && l && r ? '' : `${t ? 'top' : 'bottom'}-${l ? 'left' : 'right'}`; } function constrainPinnedSize(div, width) { const fyte = fyteMap.get(div); const maxWidth = window.innerWidth - 100 | 0; const triedExceeding = (width | 0) > maxWidth; width = Math.max(200, Math.min(maxWidth, width | 0)); return { w: width, h: width / fyte.cache.videoWidth * fyte.cache.videoHeight, triedExceeding, }; } function showOptions(e) { const [options] = translateHTML(` <div class=instant-youtube-options> <span> <label tl style="width: 100% !important;">Size: <select data-action=resize> <option tl value=Original>Original <option tl value="Fit to width">Fit to width <option>360p <option>480p <option>720p <option>1080p <option tl value=Custom>Custom... </select> </label> <label data-action=resize-custom ${cfg.resize !== 'Custom' ? 'disabled' : ''}> <input type=number min=320 max=9999 tl-placeholder=width data-action=width step=1> x <input type=number min=240 max=9999 tl-placeholder=height data-action=height step=1> </label> </span> <label tl=content,title title=msgStoryboardTip> <input data-action=showStoryboard type=checkbox> msgStoryboard </label> <span> <label tl=content,title title=msgDirectTip> <input data-action=playHTML5 type=checkbox> msgDirect </label> <label tl=content,title title=msgDirectTip> <input data-action=playHTML5Shown type=checkbox> msgDirectShown </label> </span> <label tl=content,title title=msgSafeTip> <input data-action=skipCustom type=checkbox> msgSafe </label> <table> <tr> <td><label tl=content,title title=msgPinningTip>msgPinning</label></td> <td> <select data-action=pinnable> <option tl value=on>msgPinningOn <option tl value=hide>msgPinningHover <option tl value=off>msgPinningOff </select> </td> </tr> </table> <span data-action=buttons> <button tl data-action=ok>OK</button> <button tl data-action=cancel>Cancel</button> </span> </div> `); for (const [k, v] of Object.entries(cfg)) { const el = $(`[data-action=${k}]`, options); if (el) el[el.type === 'checkbox' ? 'checked' : 'value'] = v; } $('[data-action=resize]', options).onchange = function () { const v = this.value !== 'Custom'; const e = $('[data-action=resize-custom]', options); e.children[0].disabled = e.children[1].disabled = v; v ? e.setAttribute('disabled', '') : e.removeAttribute('disabled'); }; $('[data-action=buttons]', options).onclick = e => { const btn = e.target; if (btn.dataset.action !== 'ok') { options.remove(); return; } let shouldAdjust; const oldCfg = Object.assign({}, cfg); for (const [k, v] of Object.entries(cfg)) { const el = $(`[data-action=${k}]`, options); const newVal = el && ( el.type === 'checkbox' ? el.checked : el.type === 'number' ? el.valueAsNumber : el.value); if (newVal != null && newVal !== v) { GM_setValue(k, newVal); cfg[k] = newVal; shouldAdjust = true; } } options.remove(); if (cfg.resize === 'Custom' && (cfg.width !== oldCfg.width || cfg.height !== oldCfg.height)) updateCustomSize(cfg.width, cfg.height); if (cfg.showStoryboard !== oldCfg.showStoryboard) $$('.instant-youtube-container').forEach(updateHoverHandler); if (cfg.playHTML5 !== oldCfg.playHTML5 && cfg.playHTML5Shown) { const alt = _(`msgPlay${cfg.playHTML5 ? '' : 'HTML5'}`); for (const e of $$('.instant-youtube-alternative')) e.textContent = alt; } if (cfg.playHTML5Shown !== oldCfg.playHTML5Shown) updateAltPlayerCSS(); if (shouldAdjust) adjustNodes(e, btn.closest('.instant-youtube-container')); }; e.target.insertAdjacentElement('afterend', options); } function updateCustomSize(w, h) { cfg.width = Math.min(9999, Math.max(320, w | 0 || cfg.width | 0)); cfg.height = Math.min(9999, Math.max(240, h | 0 || cfg.height | 0)); } function updateAltPlayerCSS() { const ALT = '.instant-youtube-alternative:not(#foo)'; styledom.textContent = styledom.textContent.split(ALT)[0] + /*language=CSS*/ `${ALT} { display: ${cfg.playHTML5Shown ? 'block' : 'none'} !important; }`; } function important(cssText, rx = /;/g) { return cssText.replace(rx, '!important;'); } function $(sel, base = document) { return base.querySelector(sel) || 0; } function $$(sel, base = document) { return [...base.querySelectorAll(sel)]; } function $create(tagCls, props, children) { const [tag, cls] = tagCls.split('.'); const el = Object.assign(document.createElement(tag), props); if (cls) el.className = `instant-youtube-${cls}`; if (props && typeof props.style === 'object') overrideCSS(el, props.style); if (children && typeof children !== 'object') children = document.createTextNode(children); if (children instanceof Node) el.appendChild(children); else if (Array.isArray(children)) el.append(...children.filter(Boolean)); return el; } function $$remove(sel, base = document) { for (const el of base.querySelectorAll(sel)) el.remove(); } function overrideCSS(el, props) { const names = Object.keys(props); el.style.cssText = el.style.cssText .replace(new RegExp(`(^|\\s|;)(${names.join('|')})(:[^;]+)`, 'gi'), '$1') .replace(/[^;]\s*$/, '$&;') .replace(/^\s*;\s*/, '') + names.map(n => `${n}:${props[n]}!important;`).join(' '); return el; } // fix dumb Firefox bug function floatPadding(node, style, dir) { const padding = style['padding' + dir]; if (padding.indexOf('%') < 0) return parseFloat(padding); return parseFloat(padding) * (parseFloat(style.width) || node.clientWidth) / 100; } async function cleanupStorage() { cleanupStorage.timer = 0; const cutoff = Date.now() - CACHE_STALE_DURATION; // TODO: remove localStorage in 2024 for (const k in localStorage) { if (k.startsWith(CACHE_PREFIX)) { try { const str = localStorage[k]; const isObj = str[0] === '{'; const val = isObj ? tryJSONparse(str) || false : str; const time = isObj ? val.lastUsed : parseInt(val, 36) * 1000; if (time > cutoff) { const res = isObj ? (val.lastUsed = new Date(time), val) : unpack(val, k.slice(CACHE_PREFIX.length)); write(res); } } catch (e) {} } } // TODO: remove GM_listValues in 2024 for (const k of GM_listValues()) if (k.startsWith('cache-')) GM_deleteValue(k); try { /** @type {IDBIndex} */ const store = await db({raw: true, write: true, index: 'lastUsed'}); const req = store.openCursor(IDBKeyRange.upperBound(new Date(cutoff))); req.onerror = console.warn; req.onsuccess = () => { const cur = /** @type {IDBCursorWithValue} */ req.r###lt; if (!cur) return; cur.delete().onerror = console.warn; cur.continue(); }; } catch (e) { console.warn(e); } await write().catch(console.warn); for (const k in localStorage) if (k.startsWith(CACHE_PREFIX)) delete localStorage[k]; } function initDB() { db = TinyIDB('FYTE', { onUpgrade() { this.r###lt .createObjectStore('data', {keyPath: 'id'}) .createIndex('lastUsed', 'u'); }, }); cleanupStorage.timer = setTimeout(cleanupStorage, 1e3); return db; } async function read(ids) { const toRead = []; const toWrite = []; for (const id of ids) { const str = localStorage[CACHE_PREFIX + id]; if (str) { toWrite.push(id); unpack(str, id); } else { toRead.push(id); } } if (toRead.length) { for (const val of await (db || initDB()).getMulti(toRead)) if (val) unpack(val); } if (toWrite.length) { if (!cleanupStorage.timer) cleanupStorage.timer = setTimeout(cleanupStorage, 1000); toWrite.forEach(write); } } function unpack(data, id) { const old = !!id; const arr = (old ? data : data.a).split('\n'); const obj = { id: id || (id = data.id), lastUsed: old ? new Date(parseInt(arr.shift(), 36) * 1000) : data.u, }; for (let j = 0; j < CACHE_PROPS.length; j++) { const [key, type] = CACHE_PROPS[j]; const v = arr[j] || ''; obj[key] = type === 0 ? parseInt(v, 36) || 0 : key === 'cover' ? getCoverUrl(id, v) || '' : v; } return (dbCache[id] = obj); } function write(data) { if (data) { if (!write.timer) write.timer = setTimeout(write, dbFlushDelay); const id = typeof data === 'object' ? data.id : data; dbCache[id] = data; dbFlush.add(id); return; } const toWrite = []; for (const id of dbFlush) { const obj = dbCache[id]; let res = ''; for (const [key, type] of CACHE_PROPS) { const v = obj[key]; // not storing array values separately to minimize byte size res += `${res ? '\n' : ''}${!v ? '' : type === 0 ? (+v).toString(36) : key === 'cover' ? v.split('?')[0].split('/').pop() : v.replace(/\n/g, ' ')}`; } toWrite.push({ a: res, u: obj.lastUsed, id, }); } dbFlush.clear(); write.timer = 0; return toWrite.length ? (db || initDB()).putMulti(toWrite) : Promise.resolve(); } function tryJSONparse(s) { try { return JSON.parse(s); } catch (e) {} } function secondsToTimeString(sec) { const h = sec / 3600 | 0; const m = (sec / 60 | 0) % 60; const s = sec % 60; return `${h ? h + ':' : ''}${h && m < 10 ? 0 : ''}${m}:${s < 10 ? 0 : ''}${s}`; } function translateHTML(html) { const tmp = $create('div', {innerHTML: html.trim().replace(/\n\s*/g, '')}); for (const node of $$('[tl]', tmp)) { for (const what of (node.getAttribute('tl') || 'content').split(',')) { let child; if (what === 'content') { for (const n of [...node.childNodes].reverse()) { if (n.nodeType === Node.TEXT_NODE && n.textContent.trim()) { child = n; break; } } } else child = node.getAttributeNode(what); if (!child) continue; const src = child.textContent; const srcTrimmed = src.trim(); const tl = src.replace(srcTrimmed, _(srcTrimmed)); if (src !== tl) child.textContent = tl; } } return [...tmp.childNodes]; } function initTL() { const tlSource = { msgWatch: { en: 'watch on Youtube', ru: 'открыть на Youtube', }, msgPlay: { en: 'Play with Youtube player', ru: 'Включить плеер Youtube', }, msgPlayHTML5: { en: 'Play directly (up to 720p)', ru: 'Включить напрямую (макс. 720p)', }, msgAltPlayerHint: { en: 'Shift-click to use alternative player', ru: 'Shift-клик для смены типа плеера', }, Options: { ru: 'Опции', }, 'Size:': { ru: 'Размер:', }, Original: { ru: 'Исходный', }, 'Fit to width': { ru: 'На всю ширину', }, 'Custom...': { ru: 'Настроить...', }, width: { ru: 'ширина', }, height: { ru: 'высота', }, msgStoryboard: { en: 'Storyboard thumbnails on hover', ru: 'Раскадровка при наведении курсора', }, msgStoryboardTip: { en: 'Show storyboard preview on mouse hover at the bottom', ru: 'Показывать миникадры при наведении мыши на низ кавер-картинки', }, msgDirect: { en: 'Play directly', ru: 'Встроенный плеер браузера', }, msgDirectTip: { en: 'Shift-click a thumbnail to use the alternative player', ru: 'Удерживайте клавишу Shift при щелчке на картинке для альтернативного плеера', }, msgDirectShown: { en: 'Show under play button', ru: 'Показывать под кнопкой ►', }, msgInvidious: { en: 'Use https://invidio.us to play videos', ru: 'Использовать https://invidio.us в плеере', }, msgSafe: { en: 'Safe (skip videos with enablejsapi=1)', ru: 'Консервативный режим', }, msgSafeTip: { en: 'Do not process customized videos with enablejsapi=1 parameter (requires page reload)', ru: 'Не обрабатывать нестандартные видео с параметром enablejsapi=1 ' + '(подействует после обновления страницы)', }, msgPinning: { en: 'Corner pinning', ru: 'Закрепление по углам', }, msgPinningTip: { en: 'Enable corner pinning controls when a video is playing.\n' + 'To restore the video click the active corner pin or the original video placeholder.', ru: 'Включить шпильки по углам для закрепления видео во время просмотра.\n' + 'Для отмены можно нажать еще раз на активированный угол или на заглушку, ' + 'где исходно было видео', }, msgPinningOn: { en: 'On', ru: 'Да', }, msgPinningHover: { en: 'On, hover a corner to show', ru: 'Да, при наведении курсора', }, msgPinningOff: { en: 'Off', ru: 'Нет', }, OK: { ru: 'ОК', }, Cancel: { ru: 'Оменить', }, }; const browserLang = navigator.language || navigator.languages && navigator.languages[0] || ''; const browserLangMajor = browserLang.replace(/-.+/, ''); const tl = {}; for (const k of Object.keys(tlSource)) { const langs = tlSource[k]; const text = langs[browserLang] || langs[browserLangMajor]; if (text) tl[k] = text; } return src => tl[src] || src; } function initPlayButton() { [playbtn] = translateHTML(` <svg class="instant-youtube-play-button"> <path fill-rule="evenodd" clip-rule="evenodd" fill="#1F1F1F" class="ytp-large-play-button-svg" d="M84.15,26.4v6.35c0,2.833-0.15,5.967-0.45,9.4c-0.133,1.7-0.267,3.117-0.4,4.25l-0.15,0.95c-0.167,0.767-0.367,1.517-0.6,2.25c-0.667,2.367-1.533,4.083-2.6,5.15c-1.367,1.4-2.967,2.383-4.8,2.95c-0.633,0.2-1.316,0.333-2.05,0.4c-0.767,0.1-1.3,0.167-1.6,0.2c-4.9,0.367-11.283,0.617-19.15,0.75c-2.434,0.034-4.883,0.067-7.35,0.1h-2.95C38.417,59.117,34.5,59.067,30.3,59c-8.433-0.167-14.05-0.383-16.85-0.65c-0.067-0.033-0.667-0.117-1.8-0.25c-0.9-0.133-1.683-0.283-2.35-0.45c-2.066-0.533-3.783-1.5-5.15-2.9c-1.033-1.067-1.9-2.783-2.6-5.15C1.317,48.867,1.133,48.117,1,47.35L0.8,46.4c-0.133-1.133-0.267-2.55-0.4-4.25C0.133,38.717,0,35.583,0,32.75V26.4c0-2.833,0.133-5.95,0.4-9.35l0.4-4.25c0.167-0.966,0.417-2.05,0.75-3.25c0.7-2.333,1.567-4.033,2.6-5.1c1.367-1.434,2.967-2.434,4.8-3c0.633-0.167,1.333-0.3,2.1-0.4c0.4-0.066,0.917-0.133,1.55-0.2c4.9-0.333,11.283-0.567,19.15-0.7C35.65,0.05,39.083,0,42.05,0L45,0.05c2.467,0,4.933,0.034,7.4,0.1c7.833,0.133,14.2,0.367,19.1,0.7c0.3,0.033,0.833,0.1,1.6,0.2c0.733,0.1,1.417,0.233,2.05,0.4c1.833,0.566,3.434,1.566,4.8,3c1.066,1.066,1.933,2.767,2.6,5.1c0.367,1.2,0.617,2.284,0.75,3.25l0.4,4.25C84,20.45,84.15,23.567,84.15,26.4z M33.3,41.4L56,29.6L33.3,17.75V41.4z"><title tl>msgAltPlayerHint</title></path> <polygon fill-rule="evenodd" clip-rule="evenodd" fill="#FFFFFF" points="33.3,41.4 33.3,17.75 56,29.6"></polygon> </svg>`); return playbtn; } /** * @param {string} dbName * @param {Object} [opts] * @param {string} [opts.store] * @param {string} [opts.index] * @param {number} [opts.timeout] * @param {number} [opts.version] * @param {(this:IDBOpenDBRequest, e:IDBVersionChangeEvent)=>void} [opts.onUpgrade] * @returns {TinyIDB} */ function TinyIDB(dbName, { store = 'data', index, version, timeout = 250, onUpgrade = function (evt) { if (store) evt.target.r###lt.createObjectStore(store); }, } = {}) { /** @typedef {TinyIDBSource | ((config?:TinyIDBConfig) => Promise<TinyIDBSource>)} TinyIDB */ /** @typedef {number|string|Date|BufferSource|IDBKeyRange} IDBKey */ /** @typedef {IDBObjectStore | IDBIndex | { * getMulti: (keys: IDBKey[]) => ?[] * putMulti: (values: Object[] | [val:?, key:IDBKey][]) => IDBKey[] * }} TinyIDBSource */ /** @typedef TinyIDBConfig * @prop {boolean} [raw] - returns the raw IDBObjectStore | IDBIndex object * @prop {boolean} [write] * @prop {string} [store] * @prop {string} [index] */ let timer; /** @type {IDBDatabase} */ let db; let dbPromise; const RW = ['add', 'clear', 'delete', 'put', 'putMulti']; const handler = { apply: proxyApply, get: (cfg, method) => proxyGet.bind(null, cfg, method), }; return new Proxy(TinyIDB, handler); function proxyApply(_, thisObj, [cfg]) { return cfg && cfg.raw ? proxyGet(cfg) : new Proxy(cfg || {}, handler); } /** * @param {TinyIDBConfig} cfg * @param {keyof IDBObjectStore | 'getMulti' | 'putMulti'} method * @param {...?[]} args * @return {Promise<?>} */ async function proxyGet(cfg, method, ...args) { if (!db) await (dbPromise || open()); const { raw, index: txIndex = index, store: txStore = store, write = RW.includes(method), } = cfg; const arg = args[0]; const isMulti = !raw && (method === 'getMulti' || method === 'putMulti'); if (isMulti) { if (!Array.isArray(arg)) throw new Error('Argument must be an array'); if (!arg.length) return []; } const tx = db.transaction(txStore, write ? 'readwrite' : 'readonly'); let source = tx.objectStore(txStore); if (timer) clearTimeout(timer); if (txIndex) source = source.index(txIndex); if (timeout) tx.oncomplete = tx.onerror = closeLater; if (raw) return source; let req, resolve, reject; const promise = new Promise((ok, ko) => (resolve = ok) && (reject = ko)); if (isMulti) { source._r###lts = []; method = method.slice(0, 3); const addKey = method === 'put' && !source.keyPath; for (let i = 0, val, len = arg.length; i < len; i++) { val = arg[i]; req = addKey ? source[method](val[0], val[1]) : source[method](val); req.onsuccess = onExecMulti; req.onerror = reject; } } else { req = source[method](...args); req.onsuccess = onExec; req.onerror = reject; } req._resolve = resolve; return promise; } function onExec() { this._resolve(this.r###lt); } function onExecMulti() { const {r###lt, _resolve: cb, source: {_r###lts: arr}} = this; arr.push(r###lt); if (cb) cb(arr); } function open() { dbPromise = new Promise((resolve, reject) => { const op = indexedDB.open(dbName, version); op.onupgradeneeded = onUpgrade; op.onsuccess = onOpened; op.onerror = reject; op._resolve = resolve; }); return dbPromise; } function onOpened() { this._resolve(db = this.r###lt); dbPromise = null; } function closeLater() { timer = setTimeout(closeNow, timeout); } function closeNow() { timer = 0; if (db) { db.close(); db = null; } } } function injectStylesIfNeeded(force) { if (!fytedom[0] && !force) return; styledom = styledom || GM_addStyle(/*language=CSS*/ ` [class^="instant-youtube-"]:not(#foo) { margin: 0; padding: 0; transform: none; } ` + important(/*language=CSS*/ ` .instant-youtube-container { contain: strict; display: block; position: relative; overflow: hidden; cursor: pointer; margin: auto; font: normal 14px/1.0 sans-serif, Arial, Helvetica, Verdana; text-align: center; background: black; break-inside: avoid-column; } .instant-youtube-container[disabled] { background: #888; } .instant-youtube-container[disabled] .instant-youtube-storyboard { display: none; } .instant-youtube-container[pinned] { box-shadow: 0 0 30px black; } .instant-youtube-container[playing] { contain: none; } .instant-youtube-wrapper { width: 100%; height: 100%; } .instant-youtube-play-button { display: block; position: absolute; width: 85px; height: 60px; left: 0; right: 0; top: 0; bottom: 0; margin: auto; } .instant-youtube-loading-spinner { display: block; position: absolute; width: 20px; height: 20px; left: 0; right: 0; top: 0; bottom: 0; margin: auto; pointer-events: none; background: url(""); } .instant-youtube-container:hover .ytp-large-play-button-svg { fill: #CC181E; } .instant-youtube-alternative { display: block; position: absolute; width: 20em; height: 20px; top: 50%; left: 0; right: 0; margin: 60px auto; border: none; text-align: center; text-decoration: none; text-shadow: 1px 1px 3px black; color: white; z-index: 8; font-weight: normal; font-size: 12px; } .instant-youtube-alternative:hover { text-decoration: underline; color: white; background: transparent; } .instant-youtube-embed { z-index: 10; background: transparent; transition: opacity .25s; } .instant-youtube-title { z-index: 20; display: block; position: absolute; width: auto; top: 0; left: 0; right: 0; padding: 7px; border: none; text-shadow: 1px 1px 2px black; text-align: center; text-decoration: none; color: white; background-color: #0008; } .instant-youtube-title strong { font: bold 14px/1.0 sans-serif, Arial, Helvetica, Verdana; } .instant-youtube-title strong::after { content: " - ${_('msgWatch')}"; font-weight: normal; margin-right: 1ex; } .instant-youtube-title span { color: white; } .instant-youtube-title span::before { content: "("; } .instant-youtube-title span::after { content: ")"; } .instant-youtube-title i::before { content: ", "; } .instant-youtube-container .instant-youtube-title i { all: unset; opacity: .5; font-style: normal; color: white; } @-webkit-keyframes instant-youtube-fadein { from { opacity: 0 } to { opacity: 1 } } @-moz-keyframes instant-youtube-fadein { from { opacity: 0 } to { opacity: 1 } } @keyframes instant-youtube-fadein { from { opacity: 0 } to { opacity: 1 } } .instant-youtube-container:not(:hover) .instant-youtube-title[hidden] { display: none; } .instant-youtube-title:hover { text-decoration: underline; } .instant-youtube-title strong { color: white; } .instant-youtube-options-button { opacity: 0.6; position: absolute; right: 0; bottom: 0; padding: 1.5ex 2ex; font-size: 11px; text-shadow: 1px 1px 2px black; color: white; } .instant-youtube-options-button:hover { opacity: 1; background: rgba(0, 0, 0, 0.5); } .instant-youtube-options { display: flex; position: absolute; right: 0; bottom: 0; padding: 1ex 1ex 2ex 2ex; flex-direction: column; align-items: flex-start; line-height: 1.5; text-align: left; opacity: 1; color: white; background: black; z-index: 999; } .instant-youtube-options * { width: auto; height: auto; font: inherit; font-size: 13px; vertical-align: middle; text-transform: none; text-align: left; border-radius: 0; text-decoration: none; color: white; background: black; } .instant-youtube-options > * { margin-top: 1ex; } .instant-youtube-options table { all: unset; display: table; } .instant-youtube-options tr { all: unset; display: table-row; } .instant-youtube-options td { all: unset; display: table-cell; padding: 2px; } .instant-youtube-options label > * { display: inline; } .instant-youtube-options select { padding: .5ex .25ex; border: 1px solid #444; -webkit-appearance: menulist; } .instant-youtube-options [data-action="resize-custom"] input { width: 9ex; padding: .5ex .5ex .4ex; border: 1px solid #666; } .instant-youtube-options [data-action="buttons"] { margin-top: 1em; } .instant-youtube-options button { margin: 0 1ex 0 0; padding: .5ex 2ex; border: 2px solid gray; font-weight: bold; } .instant-youtube-options button:hover { border-color: white; } .instant-youtube-options label[disabled] { opacity: 0.25; } .instant-youtube-storyboard { height: 33%; max-height: 90px; display: block; position: absolute; left: 0; right: 0; bottom: 0; overflow: visible; transition: background-color .5s .25s; } .instant-youtube-storyboard[data-loaded]:hover { background-color: #0004; } .instant-youtube-storyboard div { display: block; position: absolute; bottom: 0; pointer-events: none; border: 3px solid #888; box-shadow: 2px 2px 10px black; transition: opacity .25s ease; background-color: transparent; background-origin: content-box; opacity: 0; } .instant-youtube-storyboard div::after { content: attr(data-time); opacity: .5; color: #fff; background-color: #000; font-weight: bold; font-size: 10px; position: absolute; bottom: 4px; left: 4px; padding: 1px 3px; } .instant-youtube-storyboard:hover div { opacity: 1; } .instant-youtube-thumbnail { all: initial; width: 100%; height: 100%; max-height: 100%; overflow: hidden; object-fit: cover; transition: opacity .5s ease-out; } .instant-youtube-container [pin] { display: block; position: absolute; width: 0; height: 0; top: auto; bottom: auto; left: auto; right: auto; border-style: solid; transition: opacity 2.5s ease-in, opacity 0.4s ease-out; opacity: 0; z-index: 100; } .instant-youtube-container[playing]:hover [pin]:not([transparent]) { opacity: 1; } .instant-youtube-container[playing] [pin]:hover { cursor: alias; opacity: 1; transition: opacity 0s; } .instant-youtube-container [pin=top-left][active] { border-top-color: green; } .instant-youtube-container [pin=top-left]:hover { border-top-color: #fc0; } .instant-youtube-container [pin=top-left] { top: 0; left: 0; border-width: 10px 10px 0 0; border-color: red transparent transparent transparent; } .instant-youtube-container [pin=top-left][transparent] { border-width: 10px 10px 0 0; } .instant-youtube-container [pin=top-right][active] { border-right-color: green; } .instant-youtube-container [pin=top-right]:hover { border-right-color: #fc0; } .instant-youtube-container [pin=top-right] { top: 0; right: 0; border-width: 0 10px 10px 0; border-color: transparent red transparent transparent; } .instant-youtube-container [pin=top-right][transparent] { border-width: 0 10px 10px 0; } .instant-youtube-container [pin=bottom-right][active] { border-bottom-color: green; } .instant-youtube-container [pin=bottom-right]:hover { border-bottom-color: #fc0; } .instant-youtube-container [pin=bottom-right] { bottom: 0; right: 0; border-width: 0 0 10px 10px; border-color: transparent transparent red transparent; } .instant-youtube-container [pin=bottom-right][transparent] { border-width: 0 0 10px 10px; } .instant-youtube-container [pin=bottom-left][active] { border-left-color: green; } .instant-youtube-container [pin=bottom-left]:hover { border-left-color: #fc0; } .instant-youtube-container [pin=bottom-left] { bottom: 0; left: 0; border-width: 10px 0 0 10px; border-color: transparent transparent transparent red; } .instant-youtube-container [pin=bottom-left][transparent] { border-width: 10px 0 0 10px; } .instant-youtube-dragndrop-placeholder { z-index: 999999999; background: rgba(0, 255, 0, 0.1); border: 2px dotted green; box-sizing: border-box; pointer-events: none; } .instant-youtube-container [size-gripper] { width: 0; position: absolute; top: 0; bottom: 0; cursor: e-resize; border-color: rgba(50,100,255,0.5); border-width: 12px; background: rgba(50,100,255,0.2); z-index: 99; opacity: 0; transition: opacity .1s ease-in-out, border-color .1s ease-in-out; } .instant-youtube-container[pinned*="right"] [size-gripper] { border-style: none none none solid; left: -4px; } .instant-youtube-container[pinned*="left"] [size-gripper] { border-style: none solid none none; right: -4px; } .instant-youtube-container [size-gripper]:hover { opacity: 1; } .instant-youtube-container [size-gripper]:active { opacity: 1; width: auto; left: -4px; right: -4px; } .instant-youtube-container [size-gripper][tried-exceeding] { border-color: rgba(255,0,0,0.5); } .instant-youtube-container [size-gripper][saveAs="global"] { border-color: rgba(0,255,0,0.5); } .instant-youtube-container [size-gripper][saveAs="site"] { border-color: rgba(0,255,255,0.5); } .instant-youtube-container [size-gripper][saveAs="reset"] { border-color: rgba(255,255,0,0.5); } .instant-youtube-container [size-gripper][saveAs="cancel"] { border-color: rgba(255,0,255,0.25); } .instant-youtube-container [size-gripper] > div { white-space: nowrap; color: white; font-weight: normal; line-height: 1.25; text-align: left; position: absolute; top: 50%; padding: 1ex 1em 1ex; background-color: rgba(80,150,255,0.5); } .instant-youtube-container [size-gripper] [save-as="site"] { font-weight: bold; color: yellow; } .instant-youtube-container[pinned*="left"] [size-gripper] > div { right: 0; } `, /;\n/g).replace(/(::\w+)?\s+{/g, ':not(#_)$1 {')); // move our rules to the end of HEAD to increase CSS specificity if (styledom.nextElementSibling && document.head) document.head.appendChild(styledom); updateAltPlayerCSS(); }