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

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

// ==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);
    }

})();