/* Youtube subtitles under video frame: Move youtube subtitles under video frame. Copyright (C) 2021 T1mL3arn This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ // ==UserScript== // @name Youtube subtitles under video frame // @name:RU Субтитры Youtube под видео // @description Have you ever been annoyed by youtube subtitles covering some important part of the video? No more! The userscript moves subtitles under video frame (but you can still drag-move them horizontally). It works for default and theater modes. // @description:RU Вам когда-нибудь мешали субтитры Youtube, закрывыющие какую-то важную область видео? Пора это прекратить! Этот скрипт сдвигает субтитры под видео (вы все еще можете перетаскивать их по горизонтали). Работает в режимах "обычный" и "широкий экран". // @namespace https://github.com/t1ml3arn-userscript-js // @version 1.0.1 // @match https://www.youtube.com/* // @grant none // @run-at document-idle // @author T1mL3arn // @icon  // @homepageURL https://github.com/t1ml3arn-userscript-js/Youtube-subtitles-under-video-frame // @supportURL https://github.com/t1ml3arn-userscript-js/Youtube-subtitles-under-video-frame/issues // @license GPLv3 // @downloadURL none // ==/UserScript== const css = ` .yfms-userjs:not([fullscreen]) .caption-window.ytp-caption-window-bottom { margin-bottom: 0 !important; margin-top: 0 !important; position: absolute !important; bottom: 0 !important; top: calc(100% + 16px) !important; z-index: 9999 !important; } .yfms-userjs .html5-video-player { /* to make subs visible when they are outside player frame */ overflow: visible; /* to make player to be on top (combined with captions z-index rule, it places captions over any element on the page) */ z-index: 999; } ytd-watch-flexy #info.ytd-watch-flexy { transition: margin-top 0.25s; } .yfms-userjs #info.ytd-watch-flexy { margin-top: 64px; transition: margin-top 0.25s; } .yfms-userjs[theater] #info.ytd-watch-flexy { margin-top: 84px; } ` const SUBS_BUTTON_SELECTOR = '.ytp-subtitles-button' const USERJS_ELT_CLASS = 'yfms-userjs' const USERJS_STYLE_ID = 'youtube-subs-under-video-css' function waitElem(selector, interval = 500, timeout = 10000) { const e = document.querySelector(selector) if (e) return Promise.resolve(e) else return new Promise((resolve, reject) => { let timerId, timeoutId; timerId = setInterval(() => { const e = document.querySelector(selector) if (e) { clearInterval(timerId) clearTimeout(timeoutId) resolve(e) } }, interval); timeoutId = setTimeout(() => { clearInterval(timerId) clearTimeout(timeoutId) reject(`Cannot find element "${selector}" (timeout)`) }, timeout); }) } function addStyles() { const style = document.head.appendChild(document.createElement('style')) style.textContent = css; style.id = USERJS_STYLE_ID } function switchSubtitlesMarker(subsButton) { const pressed = subsButton.getAttribute('aria-pressed') // this elt gets special attribute by youtube when view mode changes, // so it also gets my marker class to apply my CSS const playerElt = document.querySelector('ytd-watch-flexy') if (pressed === 'true') playerElt.classList.add(USERJS_ELT_CLASS) else if (pressed === 'false') playerElt.classList.remove(USERJS_ELT_CLASS) } const init = async () => { 'use strict'; let subsButton; try { subsButton = await waitElem(SUBS_BUTTON_SELECTOR, 100, 4000) } catch (e) { console.info(`Video ${window.location.href} has no subtitles`); return } switchSubtitlesMarker(subsButton) subsButton.addEventListener('click', e => { // if my styles are not added - add them if (!document.getElementById(USERJS_STYLE_ID)) { addStyles() } switchSubtitlesMarker(subsButton) }) } init();