您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
适用于所有视频播放器的截图脚本(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); } })();