YouTube Helper

YouTube 1.Loop playback of YouTube videos 2.screenshot download 3.themed progress bar.

Versión del día 14/3/2025. Echa un vistazo a la versión más reciente.

Tendrás que instalar una extensión para tu navegador como Tampermonkey, Greasemonkey o Violentmonkey si quieres utilizar este script.

You will need to install an extension such as Tampermonkey to install this script.

Necesitarás instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Userscripts para instalar este script.

Necesitará instalar una extensión como Tampermonkey para instalar este script.

Necesitarás instalar una extensión para administrar scripts de usuario si quieres instalar este script.

(Ya tengo un administrador de scripts de usuario, déjame instalarlo)

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión como Stylus para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

Necesitará instalar una extensión del gestor de estilos de usuario para instalar este estilo.

(Ya tengo un administrador de estilos de usuario, déjame instalarlo)

// ==UserScript==
// @name         YouTube  Helper
// @name:zh-CN   YouTube 小助手
// @description  YouTube  1.Loop playback of YouTube videos 2.screenshot download 3.themed progress bar.
// @description:zh-CN YouTube 1.视频循环播放 2.截图下载 3.主题进度条
// @author       Carokahn,bernzrdo,FunnyMonkey,人民的勤务员 <[email protected]>
// @namespace    https://github.com/ChinaGodMan/UserScripts
// @supportURL   https://github.com/ChinaGodMan/UserScripts/issues
// @homepageURL  https://github.com/ChinaGodMan/UserScripts
// @license      MIT
// @icon         https://www.youtube.com/s/desktop/ee47b5e0/img/logos/favicon_144x144.png
// @match       https://www.youtube.com/*
// @match       https://m.youtube.com/*
// @compatible   chrome
// @compatible   firefox
// @compatible   edge
// @compatible   opera
// @compatible   safari
// @compatible   kiwi
// @compatible   qq
// @compatible   via
// @compatible   brave
// @version      2025.03.15.0436
// @grant        GM_addStyle
// @created      2025-03-14 20:36:01
// @modified     2025-03-14 20:36:01
// ==/UserScript==
/**
 * File: youtube-helper.user.js
 * Project: UserScripts
 * File Created: 2025/03/15,Saturday 04:36:02
 * Author: 人民的勤务员@ChinaGodMan ([email protected])
 * -----
 * Last Modified: 2025/03/15,Saturday 05:57:23
 * Modified By: 人民的勤务员@ChinaGodMan ([email protected])
 * -----
 * License: MIT License
 * Copyright © 2024 - 2025 ChinaGodMan,Inc
 */
const directDownload = true
const infiniteLool = true





const loopVideo = () => {
    const video = document.querySelector('video')
    if (video && !video.loop) {
        video.loop = true
    }
}
const ThemeProgressbar = () => {
    const css_248z = '.html5-play-progress,.ytp-play-progress{background:url("") repeat-x!important;background:linear-gradient(180deg,red 0,red 16.5%,#f90 0,#f90 33%,#ff0 0,#ff0 50%,#3f0 0,#3f0 66%,#09f 0,#09f 83.5%,#63f 0,#63f)!important;background:-webkit-linear-gradient(top,red,red 16.5%,#f90 0,#f90 33%,#ff0 0,#ff0 50%,#3f0 0,#3f0 66%,#09f 0,#09f 83.5%,#63f 0,#63f)!important;background:-moz-linear-gradient(top,red 0,red 16.5%,#f90 16.5%,#f90 33%,#ff0 33%,#ff0 50%,#3f0 50%,#3f0 66%,#09f 66%,#09f 83.5%,#63f 83.5%,#63f 100%)!important}.html5-load-progress,.ytp-load-progress{background:url("")!important}.html5-scrubber-button,.ytp-scrubber-button{background:url("")!important;border:none!important;height:21px!important;margin-left:-18px!important;margin-top:0!important;transform:scale(.8);-webkit-transform:scale(.8);-moz-transform:scale(.8);-ms-transform:scale(.8);width:34px!important}.ytp-progress-bar-container:hover .ytp-load-progress,.ytp-progress-bar-container:hover .ytp-scrubber-button{image-rendering:pixelated}.html5-progress-bar-container,.ytp-progress-bar-container{height:12px!important}.html5-progress-bar,.ytp-progress-bar{margin-top:12px!important}.html5-progress-list,.video-ads .html5-progress-list.html5-ad-progress-list,.video-ads .ytp-progress-list.ytp-ad-progress-list,.ytp-progress-list{height:12px!important}.ytp-volume-slider-track{background:#0c4177!important}'
    GM_addStyle(css_248z)
}

let escapeHTMLPolicy = 'trustedTypes' in window
    ? trustedTypes.createPolicy('forceInner', { createHTML: html => html })
    : { createHTML: html => html }
function screenBtnUpdate() {
    let $miniplayerBtn = document.querySelector('button.ytp-miniplayer-button')
    if ($miniplayerBtn && !document.getElementById('ytp-screenshot-button')) {
        const $btn = document.createElement('button')
        $btn.id = 'ytp-screenshot-button'
        $btn.classList.add('ytp-screenshot-button', 'ytp-button')
        $btn.dataset.priority = '5'
        $btn.dataset.tooltipTargetId = 'ytp-screenshot-button'
        $btn.dataset.titleNoTooltip = 'Screenshot'
        $btn.ariaLabel = 'Screenshot'
        $btn.title = 'Screenshot'
        $btn.innerHTML = escapeHTMLPolicy.createHTML(`<svg height="100%" version="1.1" viewBox="-300 -1260 1560 1560" width="100%">
            <use class="ytp-svg-shadow" xlink:href="#ytp-id-screenshot-svg"></use>
            <path
                d="M200-120q-33 0-56.5-23.5T120-200v-560q0-33 23.5-56.5T200-840h560q33 0 56.5 23.5T840-760v560q0 33-23.5 56.5T760-120H200Zm0-80h560v-560H200v560Zm40-80h480L570-480 450-320l-90-120-120 160Zm-40 80v-560 560Z"
                fill="#fff" id="ytp-id-screenshot-svg"></path>
        </svg>`)
        $btn.addEventListener('click', screenshot)

        insertBefore($btn, $miniplayerBtn)
    }

    requestAnimationFrame(screenBtnUpdate)
}
function insertBefore($element, $sibling) {
    $sibling.parentElement.insertBefore($element, $sibling)
}
function screenshot() {

    const $video = document.querySelector('#player video')
    if (!$video) {
        console.error('No video found to screenshot!')
        return
    }

    let wasPlaying = !$video.paused
    if (wasPlaying) $video.pause()

    const $canvas = document.createElement('canvas')
    $canvas.width = $video.videoWidth
    $canvas.height = $video.videoHeight

    let oldCss = $video.style.cssText
    $video.style.width = $video.videoWidth + 'px'
    $video.style.height = $video.videoHeight + 'px'

    const ctx = $canvas.getContext('2d')
    ctx.drawImage($video, 0, 0, $video.videoWidth, $video.videoHeight)

    $canvas.toBlob(blob => {

        if (directDownload) {
            const a = document.createElement('a')
            a.href = URL.createObjectURL(blob)
            a.download = `${getFileName()}.png`
            a.click()
        } else {
            const item = new ClipboardItem({ 'image/png': blob })
            navigator.clipboard.write([item])
        }

        $video.style.cssText = oldCss
        $canvas.remove()
        if (wasPlaying) $video.play()

    })

}
function getFileName() {
    const safeFileName = document.title.replace(/[\\/:*?"<>|]/g, '').replace(' - YouTube', '')
    return safeFileName
}
if (infiniteLool) {
    const observer = new MutationObserver(loopVideo)
    observer.observe(document.body, { childList: true, subtree: true })
}

requestAnimationFrame(screenBtnUpdate)
ThemeProgressbar()