// ==UserScript==
// @name ctrl+单击图片自动下载超级增强版
// @namespace https://www.uiwow.com/
// @version 1.1
// @description 在图片对象上面:特别优化了Chrome浏览器下的点击拦截,不会点击后图片放大。1、按ctrl单击图片自动下载(默认转换为JPG格式);2、按alt 单击图片会弹窗选择命名方式(无需刷新网页即可生效),文件名可以根据当前日期时间time、网页域名domain、网页标题title,或用户自定义进行命名。
// @author Techwb.cn
// @match *://*/*
// @grant GM_download
// @grant GM_setValue
// @grant GM_getValue
// @license MIT
// ==/UserScript==
(function() {
'use strict';
var clickedImage = null;
var nameOption = GM_getValue('nameOption', 'time'); //默认命名方式time
var customName = GM_getValue('customName', '');
var isDownloading = false;
// 检测Chrome浏览器
var isChrome = /Chrome/.test(navigator.userAgent) && /Google Inc/.test(navigator.vendor);
// 关键改进:使用多个事件阶段和类型进行拦截
document.addEventListener('mousedown', handleMouseDown, true);
document.addEventListener('click', handleClick, true);
document.addEventListener('mouseup', handleMouseUp, true);
// 拦截右键菜单,防止图片被打开
if (isChrome) {
document.addEventListener('contextmenu', function(e) {
if (isDownloading && e.target.tagName === 'IMG') {
e.preventDefault();
e.stopImmediatePropagation();
}
}, true);
}
function handleMouseDown(event) {
var target = event.target;
if (isImageElement(target) && event.button === 0) {
clickedImage = target;
if (event.ctrlKey) {
// 标记正在下载
isDownloading = true;
// 彻底阻止事件传播
event.stopImmediatePropagation();
event.preventDefault();
// 特殊处理Chrome浏览器
if (isChrome) {
// 临时修改图片属性,防止浏览器打开
var originalSrc = clickedImage.src;
clickedImage.src = '';
// 延迟执行下载,确保属性修改生效
setTimeout(function() {
handleDownload(originalSrc);
// 恢复图片源(可选)
setTimeout(function() {
clickedImage.src = originalSrc;
isDownloading = false;
}, 100);
}, 0);
} else {
handleDownload();
}
} else if (event.altKey) {
event.preventDefault();
handleNaming();
}
}
}
function handleClick(event) {
// 确保Ctrl+点击事件不会触发其他点击处理
if (isDownloading && event.button === 0) {
event.stopImmediatePropagation();
event.preventDefault();
}
}
function handleMouseUp(event) {
// 重置下载状态
if (isDownloading && event.button === 0) {
setTimeout(function() {
isDownloading = false;
}, 500);
}
}
// 增强版:判断元素是否是图片或图片容器
function isImageElement(element) {
if (element.tagName === 'IMG') return true;
// 检查是否是常见的图片查看器容器
if (element.classList.contains('fancybox-image') ||
element.classList.contains('magnifier-image') ||
element.classList.contains('zoom-img') ||
element.classList.contains('photo-gallery-image')) {
return true;
}
// 检查内部是否包含图片
var img = element.querySelector('img');
if (img && element.getBoundingClientRect().width === img.getBoundingClientRect().width) {
return true;
}
return false;
}
function handleDownload(forcedSrc) {
if (!clickedImage) return;
// 使用强制提供的src(如果有)
var imgSrc = forcedSrc || getRealImageUrl(clickedImage);
if (!imgSrc) {
console.log('无法获取图片URL');
return;
}
// 尝试使用GM_download,失败则回退到原生方法
try {
// 先获取原始文件名
var fileName = getFileName(imgSrc);
// 如果是GIF,进一步检查是否为动画
if (fileName.toLowerCase().endsWith('.gif')) {
checkIfAnimatedGif(imgSrc).then(isAnimated => {
// 如果不是动画且用户偏好其他格式,可以考虑转换
// 这里保持原有行为,只确保正确的扩展名
downloadWithName(imgSrc, fileName);
}).catch(err => {
console.log('GIF动画检测失败,使用默认文件名:', err);
downloadWithName(imgSrc, fileName);
});
} else {
// 非GIF图片直接下载
downloadWithName(imgSrc, fileName);
}
} catch (e) {
console.log('下载出错,回退到原生方法:', e);
fallbackDownload(imgSrc, getFileName(imgSrc));
}
}
function downloadWithName(imgSrc, fileName) {
if (typeof GM_download !== 'undefined') {
GM_download({
url: imgSrc,
name: fileName,
onerror: function(err) {
console.log('GM_download失败,回退到原生方法:', err);
fallbackDownload(imgSrc, fileName);
}
});
} else {
fallbackDownload(imgSrc, fileName);
}
}
// 原生下载方法,增强了对不同URL类型的处理
function fallbackDownload(imgSrc, fileName) {
// 处理dataURL
if (imgSrc.startsWith('data:')) {
var a = document.createElement('a');
a.href = imgSrc;
a.download = fileName;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
return;
}
// 处理普通URL
fetch(imgSrc, { mode: 'cors' })
.then(function(response) {
if (!response.ok) throw new Error('网络响应错误: ' + response.status);
return response.blob();
})
.then(function(blob) {
var url = URL.createObjectURL(blob);
var a = document.createElement('a');
a.href = url;
a.download = fileName;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
URL.revokeObjectURL(url);
})
.catch(function(error) {
console.log('下载失败:', error);
alert('下载失败: ' + error.message);
});
}
// 增强版:获取真实图片URL,处理各种情况
function getRealImageUrl(imgElement) {
// 优先使用data-src等可能的真实图片源
var imgSrc = imgElement.dataset.src ||
imgElement.dataset.original ||
imgElement.dataset.lazySrc ||
imgElement.dataset.image ||
imgElement.src;
// 处理图片代理URL
if (imgSrc.includes('proxy?url=')) {
imgSrc = decodeURIComponent(imgSrc.split('proxy?url=')[1]);
}
// 确保URL有效
if (!imgSrc || imgSrc.startsWith('data:')) {
return imgSrc;
}
// 添加协议头(如果缺失)
if (imgSrc.startsWith('//')) {
imgSrc = window.location.protocol + imgSrc;
}
return imgSrc;
}
// 关键改进:智能识别图片格式,保持GIF不变
function getFileName(imgUrl) {
var fileName = '';
if (nameOption === 'time') {
fileName = '图片_' + getDate();
} else if (nameOption === 'domain') {
fileName = getDomain();
} else if (nameOption === 'title') {
fileName = getTitle();
} else {
fileName = customName;
}
// 从URL中提取文件扩展名
var ext = getFileExtension(imgUrl);
// 过滤非法文件名字符
fileName = fileName.replace(/[\\/:*?"<>|]/g, '_');
// 添加扩展名(如果有)
return ext ? fileName + '.' + ext : fileName + '.jpg';
}
// 关键改进:获取并保留原始图片格式
function getFileExtension(url) {
if (!url) return '';
// 尝试从URL中提取扩展名
var ext = url.split('?')[0].split('.').pop().toLowerCase();
// 检查是否是有效的图片扩展名
var validExtensions = ['jpg', 'jpeg', 'png', 'gif', 'webp', 'bmp', 'svg'];
if (validExtensions.includes(ext)) {
return ext;
}
// 如果无法确定,尝试从Content-Type判断(针对dataURL)
if (url.startsWith('data:')) {
var matches = url.match(/data:image\/([^;]+);/);
if (matches && matches[1]) {
return matches[1].toLowerCase();
}
}
// 默认返回jpg(保持原有行为)
return '';
}
// 新增:检测GIF是否为动画
function checkIfAnimatedGif(url) {
return new Promise((resolve, reject) => {
// 对于dataURL直接处理
if (url.startsWith('data:')) {
checkGifData(url).then(resolve).catch(reject);
return;
}
// 对于普通URL,先获取二进制数据
fetch(url)
.then(response => response.arrayBuffer())
.then(buffer => {
checkGifBuffer(buffer).then(resolve).catch(reject);
})
.catch(reject);
});
}
// 检测GIF数据是否包含多个帧
function checkGifBuffer(buffer) {
return new Promise((resolve, reject) => {
try {
// 检查GIF签名 (GIF87a 或 GIF89a)
const signature = new TextDecoder().decode(buffer.slice(0, 6));
if (signature !== 'GIF87a' && signature !== 'GIF89a') {
resolve(false);
return;
}
// 查找动画帧标记 (0x2C)
const view = new DataView(buffer);
let offset = 13; // 跳过头部
// 查找图像分隔符 (0x2C)
let frameCount = 0;
while (offset < view.byteLength) {
const code = view.getUint8(offset);
if (code === 0x2C) { // 图像分隔符
frameCount++;
if (frameCount > 1) {
resolve(true); // 发现多个帧,是动画
return;
}
// 跳过图像数据
offset += 9; // 跳到图像宽度字段
offset += 2; // 跳过宽度
offset += 2; // 跳过高度
offset += 1; // 跳过标志位
// 跳过局部颜色表
const flags = view.getUint8(offset - 1);
if (flags & 0x80) { // 有局部颜色表
const colorTableSize = (flags & 0x07) + 1;
offset += 3 * Math.pow(2, colorTableSize);
}
// 跳过图像数据
while (offset < view.byteLength) {
const blockSize = view.getUint8(offset);
if (blockSize === 0) break;
offset += blockSize + 1;
}
offset++; // 跳过块终止符
} else if (code === 0x21) { // 扩展块
offset += 1; // 跳过扩展标签
// 跳过扩展块
while (offset < view.byteLength) {
const blockSize = view.getUint8(offset);
if (blockSize === 0) break;
offset += blockSize + 1;
}
offset++; // 跳过块终止符
} else {
break; // 不是有效的GIF格式
}
}
resolve(frameCount > 1); // 如果只有一帧,不是动画
} catch (error) {
console.error('GIF解析错误:', error);
resolve(false); // 解析失败,保守地认为不是动画
}
});
}
// 处理dataURL格式的GIF
function checkGifData(dataUrl) {
return new Promise((resolve, reject) => {
try {
// 提取base64部分
const base64 = dataUrl.split(',')[1];
if (!base64) {
resolve(false);
return;
}
// 转换为ArrayBuffer
const binary = atob(base64);
const buffer = new ArrayBuffer(binary.length);
const view = new Uint8Array(buffer);
for (let i = 0; i < binary.length; i++) {
view[i] = binary.charCodeAt(i);
}
// 检查GIF数据
checkGifBuffer(buffer).then(resolve).catch(reject);
} catch (error) {
console.error('GIF dataURL解析错误:', error);
resolve(false);
}
});
}
// 以下是原脚本的其他功能,保持不变...
function handleNaming() {
var dateTime = getDate();
var domainName = getDomain();
var titleName = getTitle();
var defaultName = dateTime + '' + titleName + '' + domainName + '.jpg';
var optionsHtml = '<option value="time" ' + (nameOption === 'time' ? 'selected' : '') + '>时间命名方式</option>' +
'<option value="domain" ' + (nameOption === 'domain' ? 'selected' : '') + '>域名命名方式</option>' +
'<option value="title" ' + (nameOption === 'title' ? 'selected' : '') + '>标题命名方式</option>' +
'<option value="custom" ' + (nameOption === 'custom' ? 'selected' : '') + '>自定义命名方式</option>';
var html = '<div><label for="nameOption">命名方式:</label><select id="nameOption">' + optionsHtml + '</select></div>';
var customNameHtml = '<div style="display:none;">' + // 初始隐藏
'<label for="customName">请输入文件名:</label>' +
'<input type="text" id="customName" value="' + customName + '">' +
'</div>';
var fileNameHtml = '<div><label for="fileName">文件名:</label><input type="text" id="fileName" value="' + defaultName + '" readonly></div>';
var dialog = document.createElement('div');
dialog.innerHTML = html + customNameHtml + fileNameHtml;
dialog.style.position = 'fixed';
dialog.style.top = '50%';
dialog.style.left = '50%';
dialog.style.transform = 'translate(-50%, -50%)';
dialog.style.backgroundColor = '#fff';
dialog.style.padding = '20px';
dialog.style.borderRadius = '5px';
dialog.style.boxShadow = '0px 0px 10px rgba(0, 0, 0, 0.5)';
dialog.style.zIndex = '9999';
document.body.appendChild(dialog);
var select = dialog.querySelector('#nameOption');
var customNameInput = dialog.querySelector('#customName');
var fileNameInput = dialog.querySelector('#fileName');
var closeButton = document.createElement('button');
closeButton.innerText = '关闭';
closeButton.style.position = 'absolute';
closeButton.style.top = '2px';
closeButton.style.right = '2px';
closeButton.style.padding = '3px 5px';
closeButton.style.backgroundColor = '#e74c3c';
closeButton.style.border = 'none';
closeButton.style.color = '#fff';
closeButton.style.borderRadius = '3px';
closeButton.style.cursor = 'pointer';
closeButton.addEventListener('click', function() {
document.body.removeChild(dialog);
});
dialog.appendChild(closeButton);
select.addEventListener('change', function() {
nameOption = select.value;
updateFileName();
GM_setValue('nameOption', nameOption);
if (nameOption === 'custom') {
customNameInput.parentNode.style.display = 'block'; // 显示
} else {
customNameInput.parentNode.style.display = 'none'; // 隐藏
GM_setValue('customName', '');
}
});
customNameInput.addEventListener('input', function() {
customName = customNameInput.value;
updateFileName();
GM_setValue('customName', customName);
});
function updateFileName() {
var fileName = '';
if (nameOption === 'time') {
fileName = '图片_' + getDate();
} else if (nameOption === 'domain') {
fileName = getDomain();
} else if (nameOption === 'title') {
fileName = getTitle();
} else {
fileName = customName;
}
// 这里默认显示为jpg,实际下载时会根据图片类型自动调整
fileNameInput.value = fileName + '.jpg';
}
updateFileName();
}
function getDate() {
var now = new Date();
return now.getFullYear() + '-' + (now.getMonth() + 1).toString().padStart(2, '0') + '-' + now.getDate().toString().padStart(2, '0') + '_' + now.getHours().toString().padStart(2, '0') + now.getMinutes().toString().padStart(2, '0') + now.getSeconds().toString().padStart(2, '0');
}
function getDomain() {
var url = window.location.href;
var domain = url.split('/')[2];
domain = domain.replace(/www\./, '');
return domain;
}
function getTitle() {
var title = document.title;
return title.substring(0, 10).replace(/[\\/:*?"<>|]/g, '_');
}
})();