// ==UserScript== // @name 图片快捷下载与查看工具 // @namespace https://scriptcat.org/ // @version 1.2 // @description 按住快捷键并点击图片即可下载或查看原图,智能识别B站等网站的原图链接 // @icon  // @author Developer // @match *://*/* // @grant GM_download // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_getValue // @grant GM_addStyle // @grant GM_notification // @grant GM_xmlhttpRequest // @grant GM_setClipboard // @connect * // ==/UserScript== (function() { 'use strict'; // 配置 let config = { hotkey: 'Alt', showPreview: true, defaultAction: 'download', showHint: true, downloadMethod: 'auto', // auto, blob, direct, clipboard // 原图识别规则 originalImageRules: { 'bilibili.com': [ { pattern: /(.+\.(jpg|jpeg|png|gif|webp))@[^?]+(?:\?.*)?$/i, replacement: '$1' }, { pattern: /(.+\.(jpg|jpeg|png|gif|webp))[?].*$/i, replacement: '$1' } ], 'weibo.com': [ { pattern: /\/thumbnail$/, replacement: '' }, { pattern: /\/small$/, replacement: '/large' } ], 'zhihu.com': [ { pattern: /_([sm]|xs|md|lg)$/, replacement: '' } ], // 通用规则 'default': [ { pattern: /(.+\.(jpg|jpeg|png|gif|webp))@[^?]+(?:\?.*)?$/i, replacement: '$1' }, { pattern: /(.+\.(jpg|jpeg|png|gif|webp))[?].*$/i, replacement: '$1' }, { pattern: /[?&](width|height|size|format|quality)=\d+/g, replacement: '' }, { pattern: /[?&]thumb(nail)?=/g, replacement: '' } ] } }; // 状态变量 let isHotkeyPressed = false; let currentSelectedImg = null; // 添加CSS样式 GM_addStyle(` :root { --primary-color: #4299e1; --success-color: #48bb78; --neutral-color: #718096; --dark-bg: #2d3748; --darker-bg: #1a202c; --header-bg: #4a5568; --text-light: #ffffff; --text-muted: #a0aec0; --border-radius: 8px; --shadow: 0 10px 25px rgba(0, 0, 0, 0.5); --transition: all 0.3s ease; } .img-selector-overlay { position: fixed !important; top: 0 !important; left: 0 !important; width: 100% !important; height: 100% !important; background: rgba(0, 0, 0, 0.5) !important; display: flex !important; justify-content: center !important; align-items: center !important; z-index: 2147483647 !important; backdrop-filter: blur(2px) !important; padding: 16px !important; box-sizing: border-box !important; } .img-selector-modal { background: var(--dark-bg) !important; border-radius: var(--border-radius) !important; box-shadow: var(--shadow) !important; width: 100% !important; max-width: 420px !important; overflow: hidden !important; color: var(--text-light) !important; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, sans-serif !important; animation: modalFadeIn 0.3s ease !important; display: flex !important; flex-direction: column !important; max-height: 90vh !important; } @keyframes modalFadeIn { from { opacity: 0; transform: translateY(-20px) scale(0.95); } to { opacity: 1; transform: translateY(0) scale(1); } } .img-selector-header { background: var(--header-bg) !important; padding: 16px !important; display: flex !important; justify-content: space-between !important; align-items: center !important; flex-shrink: 0 !important; } .img-selector-title { font-size: 18px !important; font-weight: 600 !important; margin: 0 !important; } .img-selector-close { background: none !important; border: none !important; color: var(--text-light) !important; font-size: 24px !important; cursor: pointer !important; padding: 0 !important; width: 32px !important; height: 32px !important; display: flex !important; align-items: center !important; justify-content: center !important; border-radius: 50% !important; transition: var(--transition) !important; } .img-selector-close:hover { background: rgba(255, 255, 255, 0.1) !important; } .img-selector-body { padding: 20px !important; display: flex !important; flex-direction: column !important; gap: 16px !important; overflow-y: auto !important; flex: 1 !important; } .img-preview-container { position: relative !important; width: 100% !important; background: var(--header-bg) !important; border-radius: var(--border-radius) !important; overflow: hidden !important; display: flex !important; justify-content: center !important; align-items: center !important; min-height: 150px !important; } .img-preview { width: 100% !important; max-height: 200px !important; object-fit: contain !important; border-radius: var(--border-radius) !important; } .img-info { display: flex !important; flex-direction: column !important; gap: 8px !important; font-size: 14px !important; } .img-info-row { display: flex !important; justify-content: space-between !important; align-items: center !important; flex-wrap: wrap !important; } .img-info-label { color: var(--text-muted) !important; font-weight: 500 !important; } .img-info-value { font-weight: 500 !important; text-align: right !important; max-width: 60% !important; overflow: hidden !important; text-overflow: ellipsis !important; } .img-selector-actions { display: flex !important; gap: 12px !important; justify-content: flex-end !important; margin-top: 10px !important; flex-wrap: wrap !important; } .img-selector-btn { padding: 10px 16px !important; border: none !important; border-radius: var(--border-radius) !important; font-weight: 600 !important; cursor: pointer !important; transition: var(--transition) !important; display: flex !important; align-items: center !important; gap: 8px !important; font-size: 14px !important; flex: 1 !important; min-width: 120px !important; justify-content: center !important; } .btn-download { background: var(--primary-color) !important; color: var(--text-light) !important; } .btn-download:hover { background: #3182ce !important; transform: translateY(-2px) !important; } .btn-view { background: var(--success-color) !important; color: var(--text-light) !important; } .btn-view:hover { background: #38a169 !important; transform: translateY(-2px) !important; } .btn-cancel { background: var(--neutral-color) !important; color: var(--text-light) !important; } .btn-cancel:hover { background: #4a5568 !important; transform: translateY(-2px) !important; } .img-selector-highlight { outline: 3px solid var(--primary-color) !important; outline-offset: -3px !important; transition: var(--transition) !important; position: relative !important; z-index: 2147483646 !important; } .img-selector-highlight::after { content: '' !important; position: absolute !important; top: 0 !important; left: 0 !important; right: 0 !important; bottom: 0 !important; background: rgba(66, 153, 225, 0.1) !important; pointer-events: none !important; z-index: 2147483646 !important; } .settings-panel { position: fixed !important; top: 50% !important; left: 50% !important; transform: translate(-50%, -50%) !important; background: var(--dark-bg) !important; border-radius: var(--border-radius) !important; box-shadow: var(--shadow) !important; width: 90% !important; max-width: 500px !important; max-height: 90vh !important; overflow-y: auto !important; z-index: 2147483647 !important; color: var(--text-light) !important; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, sans-serif !important; padding: 24px !important; display: flex !important; flex-direction: column !important; } .settings-header { display: flex !important; justify-content: space-between !important; align-items: center !important; margin-bottom: 20px !important; padding-bottom: 16px !important; border-bottom: 1px solid var(--header-bg) !important; } .settings-title { font-size: 20px !important; font-weight: 600 !important; margin: 0 !important; } .settings-close { background: none !important; border: none !important; color: var(--text-light) !important; font-size: 24px !important; cursor: pointer !important; padding: 4px !important; border-radius: 4px !important; transition: var(--transition) !important; } .settings-close:hover { background: rgba(255, 255, 255, 0.1) !important; } .settings-group { margin-bottom: 20px !important; } .settings-label { display: block !important; margin-bottom: 8px !important; font-weight: 500 !important; } .settings-input { width: 100% !important; padding: 12px !important; border-radius: var(--border-radius) !important; border: 1px solid var(--header-bg) !important; background: var(--darker-bg) !important; color: var(--text-light) !important; font-family: inherit !important; font-size: 14px !important; box-sizing: border-box !important; transition: var(--transition) !important; } .settings-input:focus { outline: none !important; border-color: var(--primary-color) !important; box-shadow: 0 0 0 3px rgba(66, 153, 225, 0.2) !important; } .settings-hint { font-size: 12px !important; color: var(--text-muted) !important; margin-top: 5px !important; } .settings-footer { display: flex !important; justify-content: flex-end !important; gap: 12px !important; margin-top: 24px !important; flex-wrap: wrap !important; } .settings-footer .img-selector-btn { min-width: 100px !important; flex: none !important; } .hotkey-hint { position: fixed !important; bottom: 20px !important; right: 20px !important; background: rgba(45, 55, 72, 0.9) !important; color: var(--text-light) !important; padding: 12px 16px !important; border-radius: var(--border-radius) !important; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, sans-serif !important; font-size: 14px !important; z-index: 2147483646 !important; box-shadow: 0 4px 12px rgba(0, 0, 0, 0.2) !important; transition: var(--transition) !important; max-width: 300px !important; backdrop-filter: blur(4px) !important; } .hotkey-hint.fade { opacity: 0.5 !important; } .url-debug { font-size: 12px !important; color: var(--text-muted) !important; margin-top: 5px !important; word-break: break-all !important; background: var(--darker-bg) !important; padding: 8px !important; border-radius: var(--border-radius) !important; } .url-debug-original { text-decoration: line-through !important; opacity: 0.7 !important; margin-bottom: 4px !important; } .url-debug-new { color: var(--success-color) !important; font-weight: 500 !important; } .download-status { position: fixed; top: 20px; right: 20px; background: rgba(45, 55, 72, 0.95); color: white; padding: 12px 16px; border-radius: 8px; font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; z-index: 2147483647; box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3); max-width: 300px; backdrop-filter: blur(4px); animation: slideIn 0.3s ease; } .download-status.success { border-left: 4px solid #48bb78; } .download-status.error { border-left: 4px solid #e53e3e; } .download-status.warning { border-left: 4px solid #d69e2e; } @keyframes slideIn { from { transform: translateX(100%); opacity: 0; } to { transform: translateX(0); opacity: 1; } } .download-options { display: flex; flex-direction: column; gap: 8px; margin-top: 10px; padding: 10px; background: rgba(255, 255, 255, 0.1); border-radius: 6px; } .download-option { display: flex; align-items: center; gap: 8px; cursor: pointer; padding: 6px; border-radius: 4px; transition: background 0.2s; } .download-option:hover { background: rgba(255, 255, 255, 0.1); } .download-option input[type="radio"] { margin: 0; } /* 响应式设计 */ @media (max-width: 480px) { .img-selector-modal { max-width: 100% !important; margin: 0 10px !important; } .img-selector-body { padding: 16px !important; } .img-selector-actions { flex-direction: column !important; } .img-selector-btn { width: 100% !important; } .settings-panel { width: 95% !important; padding: 16px !important; } .settings-footer { flex-direction: column !important; } .hotkey-hint { bottom: 10px !important; right: 10px !important; left: 10px !important; max-width: none !important; } } @media (max-width: 768px) { .img-info-row { flex-direction: column !important; align-items: flex-start !important; gap: 4px !important; } .img-info-value { max-width: 100% !important; text-align: left !important; } } /* 加载动画 */ .loading-spinner { display: inline-block !important; width: 20px !important; height: 20px !important; border: 3px solid rgba(255,255,255,.3) !important; border-radius: 50% !important; border-top-color: var(--text-light) !important; animation: spin 1s ease-in-out infinite !important; } @keyframes spin { to { transform: rotate(360deg); } } .preview-loading { position: absolute !important; top: 50% !important; left: 50% !important; transform: translate(-50%, -50%) !important; } `); // 初始化 function init() { loadConfig(); setupEventListeners(); // 只在首次使用时显示提示 const hintShown = GM_getValue('hint_shown', false); if (!hintShown && config.showHint) { createHotkeyHint(); GM_setValue('hint_shown', true); } registerMenuCommands(); } // 加载配置 function loadConfig() { const savedConfig = GM_getValue('img_downloader_config'); if (savedConfig) { config = { ...config, ...savedConfig }; } } // 保存配置 function saveConfig() { GM_setValue('img_downloader_config', config); } // 设置事件监听 function setupEventListeners() { // 键盘按下事件 document.addEventListener('keydown', (e) => { if (e.key === config.hotkey) { isHotkeyPressed = true; document.body.style.cursor = 'crosshair'; } }); // 键盘释放事件 document.addEventListener('keyup', (e) => { if (e.key === config.hotkey) { isHotkeyPressed = false; document.body.style.cursor = ''; removeImageHighlight(); } }); // 图片点击事件 - 使用捕获阶段以阻止默认行为 document.addEventListener('click', handleImageClick, true); // 鼠标移动事件(图片高亮) document.addEventListener('mousemove', (e) => { if (!isHotkeyPressed) return; const imgElement = findImageElement(e.target); highlightImage(imgElement); }); } // 处理图片点击事件 function handleImageClick(e) { if (!isHotkeyPressed) return; // 查找被点击的图片元素 const imgElement = findImageElement(e.target); if (imgElement) { // 阻止默认行为和事件传播 e.preventDefault(); e.stopPropagation(); e.stopImmediatePropagation(); currentSelectedImg = imgElement; showImageOptions(imgElement); // 对于B站等网站,额外添加一次阻止默认行为 setTimeout(() => { if (e.cancelable) { e.preventDefault(); } }, 0); return false; } } // 查找图片元素 - 增强版,专门处理B站评论区表情包 function findImageElement(element) { if (!element) return null; // 检查元素本身是否是图片 if (isImageElement(element)) { return element; } // 向上遍历DOM树查找图片元素 let currentElement = element; while (currentElement && currentElement !== document.body) { if (isImageElement(currentElement)) { return currentElement; } // 检查是否是表情包元素(常见于评论区) if (isEmojiElement(currentElement)) { return currentElement; } // 专门处理B站评论区表情包 if (isBilibiliCommentEmoji(currentElement)) { return currentElement; } currentElement = currentElement.parentElement; } return null; } // 判断是否是图片元素 function isImageElement(element) { // 标准的img标签 if (element.tagName === 'IMG') { return true; } // 带有背景图片的元素 const style = window.getComputedStyle(element); const backgroundImage = style.backgroundImage; if (backgroundImage && backgroundImage !== 'none') { return true; } // SVG图片 if (element.tagName === 'svg' || element.closest('svg')) { return true; } // Canvas元素 if (element.tagName === 'CANVAS') { return true; } // 排除文本元素 if (element.tagName === 'P' || element.tagName === 'SPAN' || element.tagName === 'DIV') { // 检查是否包含文本内容 const textContent = element.textContent || ''; if (textContent.trim().length > 0 && !element.querySelector('img')) { return false; } } return false; } // 判断是否是表情包元素 function isEmojiElement(element) { // 常见表情包类名和属性 const emojiClasses = ['emoji', 'face', 'biaoqing', 'expression', 'smiley', 'comment-emoji']; const emojiAttributes = ['data-emoji', 'data-face', 'data-src', 'data-url']; // 检查类名 for (const cls of emojiClasses) { if (element.classList.contains(cls)) { return true; } } // 检查属性 for (const attr of emojiAttributes) { if (element.hasAttribute(attr)) { return true; } } // 检查是否是图片但被其他元素包裹 if (element.querySelector('img')) { return true; } return false; } // 专门检测B站评论区表情包 function isBilibiliCommentEmoji(element) { // 检查是否在B站页面 if (!window.location.hostname.includes('bilibili.com')) { return false; } // 检查元素是否在评论区 const commentSection = element.closest('.bili-comment, [class*="comment"], [class*="Comment"]'); if (!commentSection) { return false; } // 检查元素特征 const isImg = element.tagName === 'IMG'; const hasEmojiClass = element.classList.contains('emoji') || element.classList.contains('face') || element.getAttribute('data-emoji') !== null; const hasEmojiSrc = element.src && ( element.src.includes('emoji') || element.src.includes('face') || element.src.includes('expression') || element.src.includes('hdslb.com/bfs/face') ); // 如果是图片且有表情包特征 if (isImg && (hasEmojiClass || hasEmojiSrc)) { return true; } // 如果元素包含表情包图片 const emojiImg = element.querySelector('img'); if (emojiImg && (emojiImg.classList.contains('emoji') || emojiImg.getAttribute('data-emoji') !== null || (emojiImg.src && emojiImg.src.includes('hdslb.com/bfs/face')))) { return true; } return false; } // 高亮图片 function highlightImage(imgElement) { removeImageHighlight(); if (imgElement) { // 使用更高优先级的类名 imgElement.classList.add('img-selector-highlight'); } } // 移除图片高亮 function removeImageHighlight() { const highlighted = document.querySelector('.img-selector-highlight'); if (highlighted) { highlighted.classList.remove('img-selector-highlight'); } } // 显示图片选项 function showImageOptions(imgElement) { // 如果已有模态框,先移除 const existingModal = document.querySelector('.img-selector-overlay'); if (existingModal) { document.body.removeChild(existingModal); } // 创建模态框 const overlay = document.createElement('div'); overlay.className = 'img-selector-overlay'; // 获取图片信息 const originalUrl = getBestImageUrl(imgElement); const processedUrl = getOriginalImageUrl(originalUrl); const imgName = getImageName(processedUrl); const imgSize = getImageSize(imgElement); overlay.innerHTML = `
图片操作
${config.showPreview ? `
预览
` : ''}
名称: ${imgName}
尺寸: ${imgSize}
类型: ${getImageType(processedUrl)}
${originalUrl !== processedUrl ? `
原始: ${truncateText(originalUrl, 50)}
处理后: ${truncateText(processedUrl, 50)}
` : ''}
`; document.body.appendChild(overlay); // 设置下载方法选择 const methodRadios = overlay.querySelectorAll('input[name="download-method"]'); methodRadios.forEach(radio => { radio.addEventListener('change', (e) => { config.downloadMethod = e.target.value; GM_setValue('img_downloader_config', config); }); }); // 添加事件监听 overlay.querySelector('.img-selector-close').addEventListener('click', () => { document.body.removeChild(overlay); }); overlay.querySelector('.btn-cancel').addEventListener('click', () => { document.body.removeChild(overlay); }); overlay.querySelector('.btn-download').addEventListener('click', async () => { const success = await downloadImage(processedUrl, imgName); if (success) { document.body.removeChild(overlay); } }); overlay.querySelector('.btn-view').addEventListener('click', () => { viewOriginalImage(processedUrl); document.body.removeChild(overlay); }); // 点击外部关闭 overlay.addEventListener('click', (e) => { if (e.target === overlay) { document.body.removeChild(overlay); } }); // ESC键关闭 const escHandler = (e) => { if (e.key === 'Escape') { document.body.removeChild(overlay); document.removeEventListener('keydown', escHandler); } }; document.addEventListener('keydown', escHandler); } // 截断文本 function truncateText(text, maxLength) { if (text.length <= maxLength) return text; return text.substring(0, maxLength) + '...'; } // 获取最佳图片URL - 增强版 function getBestImageUrl(imgElement) { // 尝试获取各种可能包含原图的属性 const attributes = [ 'data-src', 'data-original', 'data-url', 'data-src-original', 'data-original-src', 'data-source', 'data-original-url', 'src', 'href' ]; for (const attr of attributes) { const value = imgElement.getAttribute(attr); if (value && (value.startsWith('http') || value.startsWith('//'))) { return value.startsWith('//') ? 'https:' + value : value; } } // 检查背景图片 const style = window.getComputedStyle(imgElement); const backgroundImage = style.backgroundImage; if (backgroundImage && backgroundImage !== 'none') { const urlMatch = backgroundImage.match(/url\(["']?(.*?)["']?\)/); if (urlMatch && urlMatch[1]) { return urlMatch[1].startsWith('//') ? 'https:' + urlMatch[1] : urlMatch[1]; } } // 对于表情包元素,尝试从子元素获取 if (isEmojiElement(imgElement) || isBilibiliCommentEmoji(imgElement)) { const childImg = imgElement.querySelector('img'); if (childImg && childImg.src) { return childImg.src; } } // 默认返回src属性 return imgElement.src || ''; } // 获取原图URL(去除参数) function getOriginalImageUrl(url) { if (!url) return ''; const domain = getDomainFromUrl(url); let rules = config.originalImageRules.default; // 获取特定网站规则 for (const site in config.originalImageRules) { if (site !== 'default' && domain.includes(site)) { rules = config.originalImageRules[site]; break; } } let processedUrl = url; // 应用所有规则 for (const rule of rules) { // 修复:确保 pattern 是正则对象 let pattern = rule.pattern; if (typeof pattern === 'string') { try { // 默认全局和不区分大小写 pattern = new RegExp(pattern, 'gi'); } catch (e) { continue; } } if (pattern instanceof RegExp && pattern.test(processedUrl)) { processedUrl = processedUrl.replace(pattern, rule.replacement); } } // 特殊处理:B站@参数(确保移除所有@参数) if (domain.includes('bilibili') || domain.includes('hdslb')) { const atIndex = processedUrl.indexOf('@'); const questionMarkIndex = processedUrl.indexOf('?'); if (atIndex !== -1) { if (questionMarkIndex !== -1 && atIndex < questionMarkIndex) { processedUrl = processedUrl.substring(0, atIndex) + processedUrl.substring(questionMarkIndex); } else { processedUrl = processedUrl.substring(0, atIndex); } } } // 特殊处理:移除常见的尺寸和质量参数 const paramPatterns = [ /[?&](width|height|size|w|h|q|quality)=\d+/gi, /[?&](crop|resize|fit|scale)=[^&]*/gi, /[?&](format|fm)=[^&]*/gi, /[?&]thumb(nail)?=[^&]*/gi ]; for (const pattern of paramPatterns) { processedUrl = processedUrl.replace(pattern, ''); } // 处理可能产生的多余?或& processedUrl = processedUrl.replace(/(\?|&)+$/, ''); processedUrl = processedUrl.replace(/\?&/, '?'); // 如果处理后只剩下?,则移除它 if (processedUrl.endsWith('?')) { processedUrl = processedUrl.slice(0, -1); } return processedUrl || url; } // 从URL获取域名 function getDomainFromUrl(url) { try { const urlObj = new URL(url); return urlObj.hostname; } catch (e) { return ''; } } // 获取图片名称 function getImageName(url) { try { const urlObj = new URL(url); const pathname = urlObj.pathname; const filename = pathname.substring(pathname.lastIndexOf('/') + 1); // 移除可能存在的参数 const cleanFilename = filename.split('?')[0].split('#')[0]; // 如果文件名太长,截断 if (cleanFilename.length > 30) { return cleanFilename.substring(0, 27) + '...'; } return cleanFilename || 'image'; } catch (e) { return 'image'; } } // 获取图片尺寸 function getImageSize(imgElement) { if (imgElement.naturalWidth && imgElement.naturalHeight) { return `${imgElement.naturalWidth} × ${imgElement.naturalHeight}`; } if (imgElement.offsetWidth && imgElement.offsetHeight) { return `${imgElement.offsetWidth} × ${imgElement.offsetHeight}`; } return '未知'; } // 获取图片类型 function getImageType(url) { if (!url) return '未知'; // 移除参数和哈希 const cleanUrl = url.split('?')[0].split('#')[0]; const extension = cleanUrl.split('.').pop().toLowerCase(); if (extension && extension.length < 5) { return extension; } return '未知'; } // 显示下载状态提示 function showDownloadStatus(message, type = 'info', duration = 3000) { // 移除现有的状态提示 const existingStatus = document.querySelector('.download-status'); if (existingStatus) { document.body.removeChild(existingStatus); } const status = document.createElement('div'); status.className = `download-status ${type}`; status.textContent = message; document.body.appendChild(status); // 自动消失 if (duration > 0) { setTimeout(() => { if (document.body.contains(status)) { document.body.removeChild(status); } }, duration); } return status; } // 下载图片 - 增强版,包含多种方法 async function downloadImage(url, filename) { if (!url) { showDownloadStatus('无法获取图片地址', 'error'); return false; } // 根据配置选择下载方法 let success = false; switch (config.downloadMethod) { case 'blob': success = await downloadViaBlob(url, filename); break; case 'direct': success = downloadViaAnchor(url, filename); break; case 'clipboard': success = await copyToClipboard(url); break; case 'auto': default: // 自动选择:先尝试GM_download,失败后尝试其他方法 if (typeof GM_download !== 'undefined') { success = await downloadViaGm(url, filename); if (!success) { success = await downloadViaBlob(url, filename); if (!success) { success = downloadViaAnchor(url, filename); } } } else { success = await downloadViaBlob(url, filename); if (!success) { success = downloadViaAnchor(url, filename); } } break; } if (success) { showDownloadStatus(`已下载: ${filename}`, 'success'); } else { showDownloadStatus('下载失败,已复制图片链接到剪贴板', 'warning'); GM_setClipboard(url); } return success; } // 方法1: 使用GM_download async function downloadViaGm(url, filename) { return new Promise((resolve) => { try { const timestamp = new Date().getTime(); const downloadUrl = url.includes('?') ? `${url}&_=${timestamp}` : `${url}?_=${timestamp}`; GM_download({ url: downloadUrl, name: filename, onerror: function(e) { console.error('GM_download失败:', e.error); resolve(false); }, onload: function() { resolve(true); } }); } catch (e) { console.error('GM_download异常:', e); resolve(false); } }); } // 方法2: 使用Blob和对象URL async function downloadViaBlob(url, filename) { return new Promise((resolve) => { try { // 使用GM_xmlhttpRequest绕过CORS限制 GM_xmlhttpRequest({ method: "GET", url: url, responseType: "blob", onload: function(response) { try { const blob = response.response; const blobUrl = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = blobUrl; a.download = filename; a.style.display = 'none'; document.body.appendChild(a); a.click(); // 清理 setTimeout(() => { document.body.removeChild(a); URL.revokeObjectURL(blobUrl); }, 100); resolve(true); } catch (e) { console.error('Blob下载失败:', e); resolve(false); } }, onerror: function(error) { console.error('请求图片失败:', error); resolve(false); } }); } catch (e) { console.error('Blob下载异常:', e); resolve(false); } }); } // 方法3: 直接使用a标签下载 function downloadViaAnchor(url, filename) { try { const a = document.createElement('a'); a.href = url; a.download = filename; a.style.display = 'none'; document.body.appendChild(a); a.click(); document.body.removeChild(a); return true; } catch (e) { console.error('直接下载失败:', e); return false; } } // 方法4: 复制到剪贴板 async function copyToClipboard(url) { try { // 使用GM_xmlhttpRequest获取图片数据 const response = await new Promise((resolve) => { GM_xmlhttpRequest({ method: "GET", url: url, responseType: "blob", onload: resolve, onerror: () => resolve(null) }); }); if (!response) return false; // 将Blob转换为可以复制到剪贴板的数据 const blob = response.response; const items = [new ClipboardItem({ [blob.type]: blob })]; await navigator.clipboard.write(items); showDownloadStatus('图片已复制到剪贴板', 'success'); return true; } catch (e) { console.error('复制到剪贴板失败:', e); // 备用方案:只复制URL try { GM_setClipboard(url); showDownloadStatus('图片链接已复制到剪贴板', 'success'); return true; } catch (err) { console.error('复制URL失败:', err); return false; } } } // 查看原图 function viewOriginalImage(url) { if (!url) { showDownloadStatus('无法获取图片地址', 'error'); return; } window.open(url, '_blank'); } // 创建快捷键提示 function createHotkeyHint() { // 如果已有提示,先移除 const existingHint = document.querySelector('.hotkey-hint'); if (existingHint) { document.body.removeChild(existingHint); } const hint = document.createElement('div'); hint.className = 'hotkey-hint'; hint.textContent = `按住 ${config.hotkey} 并点击图片进行操作`; document.body.appendChild(hint); // 5秒后淡出提示 setTimeout(() => { hint.classList.add('fade'); }, 5000); // 点击提示可立即关闭 hint.addEventListener('click', () => { document.body.removeChild(hint); }); } // 注册菜单命令 function registerMenuCommands() { GM_registerMenuCommand('图片下载器设置', showSettingsPanel); GM_registerMenuCommand('重置提示', resetHint); GM_registerMenuCommand('测试原图识别', testOriginalImageRecognition); } // 显示设置面板 function showSettingsPanel() { const settingsPanel = document.createElement('div'); settingsPanel.className = 'settings-panel'; settingsPanel.innerHTML = `
图片下载器设置
按住此键并点击图片可进行操作
选择图片下载方式
在操作面板中显示图片预览
在页面上显示快捷键提示
`; document.body.appendChild(settingsPanel); // 添加事件监听 settingsPanel.querySelector('.settings-close').addEventListener('click', () => { document.body.removeChild(settingsPanel); }); settingsPanel.querySelector('#settings-cancel').addEventListener('click', () => { document.body.removeChild(settingsPanel); }); settingsPanel.querySelector('#settings-save').addEventListener('click', () => { // 保存设置 config.hotkey = settingsPanel.querySelector('#hotkey-select').value; config.downloadMethod = settingsPanel.querySelector('#download-method').value; config.showPreview = settingsPanel.querySelector('#show-preview').checked; config.showHint = settingsPanel.querySelector('#show-hint').checked; saveConfig(); // 更新提示 const hint = document.querySelector('.hotkey-hint'); if (hint && config.showHint) { hint.textContent = `按住 ${config.hotkey} 并点击图片进行操作`; } else if (hint && !config.showHint) { document.body.removeChild(hint); } document.body.removeChild(settingsPanel); }); // 点击外部关闭 settingsPanel.addEventListener('click', (e) => { if (e.target === settingsPanel) { document.body.removeChild(settingsPanel); } }); // ESC键关闭 const escHandler = (e) => { if (e.key === 'Escape') { document.body.removeChild(settingsPanel); document.removeEventListener('keydown', escHandler); } }; document.addEventListener('keydown', escHandler); } // 重置提示 function resetHint() { GM_setValue('hint_shown', false); createHotkeyHint(); GM_notification({ text: '提示已重置,将会再次显示', title: '重置成功', timeout: 2000 }); } // 测试原图识别功能 function testOriginalImageRecognition() { const testUrls = [ 'https://i1.hdslb.com/bfs/face/39a835c3d22e8e26a77fda9bb65ecab814128103.jpg@96w_96h_1c_1s_!web-avatar.avif', 'https://example.com/image.jpg@300w_200h_1e_1c', 'https://example.com/photo.png?width=300&height=200&quality=80', 'https://wx1.sinaimg.cn/large/002ABC.jpg', 'https://wx1.sinaimg.cn/small/002ABC.jpg', 'https://pic1.zhimg.com/50/v2-abc123_s.jpg', 'https://pic1.zhimg.com/80/v2-abc123_xs.jpg' ]; let result = '原图识别测试结果:\n\n'; testUrls.forEach(url => { const original = getOriginalImageUrl(url); result += `原始: ${url}\n处理后: ${original}\n\n`; }); // 显示结果 alert(result); } // 启动脚本 init(); })();