🎉 欢迎访问GreasyFork.Org 镜像站!本镜像站由公众号【爱吃馍】搭建,用于分享脚本。联系邮箱📮

Greasy fork 爱吃馍镜像

通用视频播放器截图工具(测试)

适用于所有视频播放器的截图脚本(H5、Flash、iframe等)

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

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

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

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

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

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

// ==UserScript==
// @name         通用视频播放器截图工具(测试)
// @icon         
// @version      2025.07.18
// @description  适用于所有视频播放器的截图脚本(H5、Flash、iframe等)
// @author       嘉友友
// @match        *://www.youtube.com/*
// @match        *://www.bilibili.com/*
// @match        *://live.bilibili.com/*
// @match        *://www.twitch.tv/*
// @match        *://live.douyin.com/*
// @match        *://live.kuaishou.com/*
// @license      GPL-3.0
// @namespace https://greasyfork.org/users/1336389
// ==/UserScript==

(function() {
    'use strict';

    // 缓存优化
    const cache = {
        videoElements: null,
        lastCacheTime: 0,
        cacheValidityTime: 2000, // 缓存2秒有效
        queryResults: new Map()
    };

    // 防抖函数
    function debounce(func, wait) {
        let timeout;
        return function executedFunction(...args) {
            const later = () => {
                clearTimeout(timeout);
                func(...args);
            };
            clearTimeout(timeout);
            timeout = setTimeout(later, wait);
        };
    }

    // 节流函数
    function throttle(func, limit) {
        let inThrottle;
        return function(...args) {
            if (!inThrottle) {
                func.apply(this, args);
                inThrottle = true;
                setTimeout(() => inThrottle = false, limit);
            }
        };
    }

    // 优化的页面检查(使用缓存)
    function isVideoPage() {
        const cacheKey = 'isVideoPage';
        if (cache.queryResults.has(cacheKey)) {
            return cache.queryResults.get(cacheKey);
        }

        const url = window.location.href.toLowerCase();
        const videoKeywords = [
            'youtube', 'bilibili', 'iqiyi', 'youku', 'douyin', 'tiktok',
            'twitch', 'kuaishou', 'huya', 'douyu', 'acfun', 'vimeo',
            'video', 'movie', 'play', 'watch', 'live', 'stream'
        ];

        const result = videoKeywords.some(keyword => url.includes(keyword)) ||
                      document.querySelector('video, embed, object, iframe[src*="video"], iframe[src*="player"]');

        cache.queryResults.set(cacheKey, result);
        // 清理缓存,避免内存泄漏
        setTimeout(() => cache.queryResults.delete(cacheKey), 5000);

        return result;
    }

    // 优化的视频元素查找(使用缓存和优化查询)
    function findVideoElements() {
        const now = Date.now();

        // 使用缓存
        if (cache.videoElements && now - cache.lastCacheTime < cache.cacheValidityTime) {
            return cache.videoElements;
        }

        // 优化选择器,按常用程度排序
        const selectors = [
            'video', // 最常用的放前面
            '.html5-main-video',
            '.video-stream',
            '.bilibili-live-player video',
            '.bpx-player-video-wrap video',
            '#movie_player video',
            '.player-area video',
            '.xgplayer video'
        ];

        const videos = [];
        const processedElements = new Set(); // 避免重复处理

        for (const selector of selectors) {
            try {
                const elements = document.querySelectorAll(selector);
                for (const element of elements) {
                    if (element.tagName === 'VIDEO' && !processedElements.has(element)) {
                        processedElements.add(element);
                        // 使用更高效的尺寸检查
                        if (element.offsetWidth > 0 && element.offsetHeight > 0) {
                            videos.push(element);
                        }
                    }
                }
                // 如果已经找到视频,优先使用第一个匹配的选择器结果
                if (videos.length > 0 && selector === 'video') break;
            } catch (e) {
                // 忽略选择器错误,继续下一个
                continue;
            }
        }

        // 缓存结果
        cache.videoElements = videos;
        cache.lastCacheTime = now;

        return videos;
    }

    // 优化的视频有效性检查(减少DOM查询)
    function isValidVideo(video) {
        // 先检查最简单的属性
        if (video.hidden) return false;

        // 使用 offsetWidth/offsetHeight 替代 getBoundingClientRect(性能更好)
        if (video.offsetWidth <= 0 || video.offsetHeight <= 0) return false;

        // 最后检查 computed style(最耗性能的)
        const style = video.currentStyle || window.getComputedStyle(video);
        return style.display !== 'none';
    }

    // 优化的视频元素截图(保持源质量)
    async function captureVideoElement(video) {
        try {
            // 提前检查,避免无用计算
            if (video.readyState < 1) {
                return { success: false, message: '视频尚未加载,请稍后再试' };
            }

            const canvas = document.createElement('canvas');
            const ctx = canvas.getContext('2d', { alpha: false }); // 禁用alpha通道提升性能

            canvas.width = video.videoWidth || video.clientWidth;
            canvas.height = video.videoHeight || video.clientHeight;

            try {
                ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
            } catch (e) {
                if (e.name === 'SecurityError') {
                    return { success: false, message: 'CORS安全限制,无法截取此视频' };
                }
                throw e;
            }

            return new Promise((resolve) => {
                canvas.toBlob(function(blob) {
                    if (!blob) {
                        resolve({ success: false, message: '生成图片失败' });
                        return;
                    }

                    downloadImage(blob, canvas.width, canvas.height);
                    resolve({
                        success: true,
                        width: canvas.width,
                        height: canvas.height
                    });
                }, 'image/png', 1.0); // 保持源质量 1.0
            });

        } catch (error) {
            return { success: false, message: '截图失败: ' + error.message };
        }
    }

    // 优化的H5视频截图
    function captureH5Video() {
        const videos = findVideoElements();

        // 优先级排序:可见 > 已加载 > 大尺寸
        const sortedVideos = videos
            .filter(video => isValidVideo(video) && video.readyState >= 1)
            .sort((a, b) => {
                const aRect = { width: a.offsetWidth, height: a.offsetHeight };
                const bRect = { width: b.offsetWidth, height: b.offsetHeight };
                return (bRect.width * bRect.height) - (aRect.width * aRect.height);
            });

        for (let video of sortedVideos) {
            // 优先选择较大尺寸的视频
            if (video.offsetWidth >= 200 && video.offsetHeight >= 150) {
                return captureVideoElement(video);
            }
        }

        // 如果没有大尺寸的,选择第一个可用的
        if (sortedVideos.length > 0) {
            return captureVideoElement(sortedVideos[0]);
        }

        return null;
    }

    // 优化的iframe查找(减少DOM查询)
    function captureIframeVideo() {
        const iframes = document.querySelectorAll('iframe[src*="video"], iframe[src*="player"], iframe[src*="youtube"], iframe[src*="bilibili"], iframe[src*="vimeo"], iframe[src*="live"]');

        const validIframes = [];
        for (const iframe of iframes) {
            if (iframe.offsetWidth > 0 && iframe.offsetHeight > 0) {
                validIframes.push({
                    element: iframe,
                    area: iframe.offsetWidth * iframe.offsetHeight
                });
            }
        }

        if (validIframes.length > 0) {
            // 选择面积最大的iframe
            validIframes.sort((a, b) => b.area - a.area);
            return captureElementArea(validIframes[0].element);
        }
        return null;
    }

    // 优化的Flash/Object查找
    function captureFlashVideo() {
        const objects = document.querySelectorAll('object[type*="flash"], object[data*="video"], embed[type*="flash"], embed[src*="video"]');

        const validObjects = [];
        for (const obj of objects) {
            if (obj.offsetWidth > 0 && obj.offsetHeight > 0) {
                validObjects.push(obj);
                break; // 找到第一个就够了
            }
        }

        if (validObjects.length > 0) {
            return captureElementArea(validObjects[0]);
        }
        return null;
    }

    // 优化的元素区域截图(保持源质量)
    function captureElementArea(element) {
        try {
            const width = Math.min(element.offsetWidth, 1920);
            const height = Math.min(element.offsetHeight, 1080);

            if (width === 0 || height === 0) {
                return { success: false, message: '播放器区域无效' };
            }

            const canvas = document.createElement('canvas');
            const ctx = canvas.getContext('2d', { alpha: false });

            canvas.width = width;
            canvas.height = height;

            // 优化绘制过程,减少绘制调用
            ctx.fillStyle = '#1a1a1a';
            ctx.fillRect(0, 0, width, height);

            ctx.strokeStyle = '#333333';
            ctx.lineWidth = 2;
            ctx.strokeRect(1, 1, width - 2, height - 2);

            // 合并文字绘制
            ctx.textAlign = 'center';
            ctx.fillStyle = '#ffffff';
            ctx.font = 'bold 28px Arial';
            ctx.fillText('🎬 视频播放器区域', width / 2, height / 2 - 40);

            ctx.font = '18px Arial';
            ctx.fillStyle = '#ffcc00';
            ctx.fillText('无法直接截取视频内容', width / 2, height / 2);

            ctx.fillStyle = '#888888';
            ctx.font = '16px Arial';
            ctx.fillText(`尺寸: ${Math.round(element.offsetWidth)} × ${Math.round(element.offsetHeight)}`, width / 2, height / 2 + 40);

            return new Promise((resolve) => {
                canvas.toBlob(function(blob) {
                    downloadImage(blob, width, height);
                    resolve({ success: true, isPlaceholder: true });
                }, 'image/png', 1.0); // 保持源质量 1.0
            });

        } catch (error) {
            return { success: false, message: '截图失败: ' + error.message };
        }
    }

    // 优化的主截图函数
    async function captureVideoFrame() {
        try {
            // 清除缓存,强制重新查找
            cache.videoElements = null;

            const h5Result = captureH5Video();
            if (h5Result) {
                const result = await h5Result;
                if (result.success) {
                    showMessage(`📸 截图成功!分辨率: ${result.width}×${result.height}`, 'success');
                    return;
                } else {
                    showMessage(result.message, 'warning');
                }
            }

            // 使用 requestAnimationFrame 优化后续操作
            await new Promise(resolve => requestAnimationFrame(resolve));

            const iframeResult = captureIframeVideo();
            if (iframeResult) {
                const result = await iframeResult;
                if (result.success) {
                    const type = result.isPlaceholder ? '播放器区域截图' : '截图';
                    showMessage(`📸 ${type}成功!`, 'success');
                    return;
                }
            }

            const flashResult = captureFlashVideo();
            if (flashResult) {
                const result = await flashResult;
                if (result.success) {
                    showMessage('📸 播放器区域截图成功!', 'success');
                    return;
                }
            }

            showMessage('未找到可截图的视频播放器!\n请等待视频加载完成后再试', 'error');
        } catch (error) {
            showMessage('截图过程出错:' + error.message, 'error');
        }
    }

    // 优化的视频信息获取
    function getVideoInfo() {
        const videos = findVideoElements();

        if (videos.length === 0) {
            return '📺 未检测到有效的视频播放器';
        }

        const infoParts = [`📺 检测到 ${videos.length} 个有效视频播放器:\n`];
        let validCount = 0;

        videos.forEach((video) => {
            const isVisible = isValidVideo(video);
            const videoWidth = video.videoWidth || 0;
            const videoHeight = video.videoHeight || 0;
            const displayWidth = video.offsetWidth;
            const displayHeight = video.offsetHeight;

            if (displayWidth > 0 && displayHeight > 0) {
                validCount++;
                const status = video.readyState >= 1 ? '✅' : '⏳';
                const visibility = isVisible ? '👁️' : '🚫';

                let info = `${validCount}. ${status}${visibility} `;

                if (videoWidth && videoHeight) {
                    info += `视频分辨率: ${videoWidth}×${videoHeight}\n`;
                    if (displayWidth !== videoWidth || displayHeight !== videoHeight) {
                        info += `   显示尺寸: ${displayWidth}×${displayHeight}\n`;
                    }
                } else {
                    info += `显示尺寸: ${displayWidth}×${displayHeight}\n`;
                }

                if (video.duration && !isNaN(video.duration) && video.duration !== Infinity) {
                    const duration = Math.round(video.duration);
                    const minutes = Math.floor(duration / 60);
                    const seconds = duration % 60;
                    info += `   时长: ${minutes}:${seconds.toString().padStart(2, '0')}\n`;
                }

                infoParts.push(info + '\n');
            }
        });

        // 优化iframe检查
        const validIframes = document.querySelectorAll('iframe[src*="video"], iframe[src*="player"], iframe[src*="youtube"], iframe[src*="bilibili"]');
        const visibleIframes = Array.from(validIframes).filter(iframe =>
            iframe.offsetWidth > 0 && iframe.offsetHeight > 0
        );

        if (visibleIframes.length > 0) {
            infoParts.push(`📱 iframe播放器: ${visibleIframes.length} 个\n`);
            visibleIframes.forEach((iframe, index) => {
                infoParts.push(`${index + 1}. 尺寸: ${iframe.offsetWidth}×${iframe.offsetHeight}\n`);
            });
        }

        if (validCount === 0 && visibleIframes.length === 0) {
            return '📺 未检测到有尺寸信息的播放器';
        }

        return infoParts.join('').trim();
    }

    // 优化的下载函数(使用 revokeObjectURL 的延迟清理)
    function downloadImage(blob, width, height) {
        const url = URL.createObjectURL(blob);
        const a = document.createElement('a');
        a.href = url;

        const now = new Date();
        const timestamp = now.getFullYear() +
            String(now.getMonth() + 1).padStart(2, '0') +
            String(now.getDate()).padStart(2, '0') + '_' +
            String(now.getHours()).padStart(2, '0') +
            String(now.getMinutes()).padStart(2, '0') +
            String(now.getSeconds()).padStart(2, '0');

        const domain = window.location.hostname.replace(/\./g, '_');
        a.download = `${domain}_video_${timestamp}.png`;

        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);

        // 延迟清理,确保下载完成
        setTimeout(() => URL.revokeObjectURL(url), 1000);
    }

    // 优化的消息显示(缓存样式)
    const messageStyles = {
        success: { bg: 'rgba(40, 167, 69, 0.95)', icon: '✅' },
        error: { bg: 'rgba(220, 53, 69, 0.95)', icon: '❌' },
        warning: { bg: 'rgba(255, 193, 7, 0.95)', icon: '⚠️' },
        info: { bg: 'rgba(52, 58, 64, 0.95)', icon: 'ℹ️' }
    };

    function showMessage(text, type = 'info') {
        const messageDiv = document.createElement('div');
        const style = messageStyles[type];

        messageDiv.textContent = `${style.icon} ${text}`;
        messageDiv.style.cssText = `
            position: fixed;
            top: 20px;
            right: 20px;
            background: ${style.bg};
            color: white;
            padding: 12px 20px;
            border-radius: 8px;
            font-size: 14px;
            font-family: -apple-system, BlinkMacSystemFont, sans-serif;
            z-index: 999999;
            box-shadow: 0 4px 20px rgba(0,0,0,0.3);
            max-width: 350px;
            word-wrap: break-word;
            white-space: pre-line;
            opacity: 0;
            transform: translateX(20px);
            transition: all 0.3s ease;
            will-change: transform, opacity;
        `;

        document.body.appendChild(messageDiv);

        // 使用 requestAnimationFrame 优化动画
        requestAnimationFrame(() => {
            messageDiv.style.opacity = '1';
            messageDiv.style.transform = 'translateX(0)';
        });

        const hideDelay = type === 'error' ? 5000 : 3500;
        setTimeout(() => {
            messageDiv.style.opacity = '0';
            messageDiv.style.transform = 'translateX(20px)';
            setTimeout(() => {
                if (document.body.contains(messageDiv)) {
                    document.body.removeChild(messageDiv);
                }
            }, 300);
        }, hideDelay);
    }

    // 优化的键盘事件监听(防抖处理)
    const debouncedCapture = debounce(captureVideoFrame, 300);
    const debouncedInfo = debounce(() => {
        const info = getVideoInfo();
        showMessage(info, 'info');
    }, 300);

    document.addEventListener('keydown', function(event) {
        // 避免在输入框中触发
        if (event.target.tagName === 'INPUT' || event.target.tagName === 'TEXTAREA') {
            return;
        }

        if (event.altKey && event.code === 'Digit1') {
            event.preventDefault();
            debouncedCapture();
        }

        if (event.altKey && event.code === 'Digit2') {
            event.preventDefault();
            debouncedInfo();
        }
    }, { passive: false });

    // 优化的初始化
    function initialize() {
        if (!isVideoPage()) return;

        const videos = findVideoElements();
        if (videos.length > 0) {
            showMessage(`🎬 截图工具已就绪\nAlt+1: 截图  Alt+2: 查看信息\n检测到 ${videos.length} 个有效播放器`, 'success');
        } else {
            showMessage('🎬 截图工具已就绪\nAlt+1: 截图  Alt+2: 查看信息', 'info');
        }
    }

    // 优化的页面变化监听(节流 + 防抖)
    let lastUrl = location.href;
    const throttledObserver = throttle(() => {
        const currentUrl = location.href;
        if (currentUrl !== lastUrl) {
            lastUrl = currentUrl;
            // 清除缓存
            cache.videoElements = null;
            cache.queryResults.clear();

            setTimeout(() => {
                if (isVideoPage()) initialize();
            }, 1500);
        }
    }, 1000);

    // 使用更精确的观察配置
    const observer = new MutationObserver(throttledObserver);
    observer.observe(document, {
        subtree: true,
        childList: true,
        attributes: false, // 不观察属性变化
        characterData: false // 不观察文本变化
    });

    // 页面卸载时清理资源
    window.addEventListener('beforeunload', () => {
        observer.disconnect();
        cache.queryResults.clear();
        cache.videoElements = null;
    });

    // 优化的页面加载检查
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', () =>
            setTimeout(initialize, 1000), { once: true });
    } else {
        setTimeout(initialize, 1000);
    }

})();