// ==UserScript==
// @name 电商详情页照片视频打包工具(淘宝、天猫、1688)
// @namespace taobao&tmall&1688 pictures downloader
// @version 0.2
// @description 下载一些电商网站网页的照片和视频
// @author AooMing -- 2025/3/7
// @icon 
// @match *://item.taobao.com/*
// @match *://detail.tmall.com/*
// @match *://detail.1688.com/*
// @grant GM_xmlhttpRequest
// @grant GM_addStyle
// @grant GM_download
// @require https://unpkg.com/[email protected]/dist/pizzip.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js
// @license GPL
// ==/UserScript==
(function () {
'use strict';
// 根据域名设置不同的标签名
let targetIds = [];
let innerWrapClass = ''; //主图所在的class,为webp格式的
let videoxContainerClass = ''; //视频所在的class
let thumbnailsClass = ''; //主图ui标签
let thumbnailClass = ''; //主图li标签
const domain = window.location.hostname;
if (domain.includes('item.taobao.com')) {
targetIds = ['content'];
innerWrapClass = 'innerWrap--tD6LdQYX';
videoxContainerClass = 'videox-container';
thumbnailsClass = 'thumbnails--v976to2t';
thumbnailClass = 'thumbnail--TxeB1sWz';
}
else if (domain.includes('detail.tmall.com')) {
targetIds = ['content'];
innerWrapClass = 'innerWrap--tD6LdQYX';
videoxContainerClass = 'videox-container';
thumbnailsClass = 'thumbnails--v976to2t';
thumbnailClass = 'thumbnail--TxeB1sWz';
}
else if (domain.includes('detail.1688.com')) {
targetIds = ['detailContentContainer'];
innerWrapClass = 'img-list-wrapper';
videoxContainerClass = 'lib-video';
thumbnailsClass = '';
thumbnailClass = '';
}
// 可以继续添加其他域名的配置
// else if (domain.includes('anotherdomain.com')) {
// targetIds = ['another-id'];
// innerWrapClass = 'another-inner-wrap-class';
// videoxContainerClass = 'another-videox-container-class';
// thumbnailsClass = 'another-thumbnails-class';
// thumbnailClass = 'another-thumbnail-class';
// }
// 添加样式
GM_addStyle(`
.start-button {
position: fixed;
top: 10px;
right: 10px;
background-color: #ff7e43;
color: white;
border: none;
padding: 10px 20px;
border-radius: 5px;
cursor: pointer;
z-index: 9999;
font-family: PingFang SC;
}
.start-button:hover {
background-color: #ff621a;
}
.message-popup {
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background-color: #ff621a;
color: white;
padding: 20px;
border: 1px solid #ccc;
border-radius: 5px;
box-shadow: 0 0 10px rgba(0, 0, 0, 0.5);
z-index: 9999;
font-size: 18px;
font-family: PingFang SC;
}
.selection-popup {
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background-color: white;
padding: 20px;
border: 1px solid #ccc;
border-radius: 5px;
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
z-index: 9999;
max-width: 80%;
max-height: 80%;
overflow: auto;
display: flex;
flex-direction: column;
}
.selection-popup h2 {
margin-top: 0;
margin-bottom: 20px;
text-align: center;
color:black;
font-size: 140%;
font-family: PingFangSC-Semibold;
}
.grid {
display: grid;
grid-template-columns: repeat(4, 1fr);
grid-gap: 10px;
flex-grow: 1;
overflow-y: auto;
}
.media-item {
position: relative;
}
.media-item img {
width: 100%;
height: max-content;
cursor: pointer;
transition: opacity 0.3s;
}
.media-item img.selected {
opacity: 0.3;
border: 3px solid #ff621a;
}
.media-item .checkmark {
display: none;
position: absolute;
top: 5px;
left: 5px;
font-size: 24px;
color: #ff621a;
z-index: 1;
}
.media-item img.selected + .checkmark {
display: block;
}
.media-item p {
margin: 5px 0;
font-size: 12px;
word-break: break-all;
}
.media-item p a {
color: #ff7e43;
text-decoration: none;
}
.media-item p a:hover {
text-decoration: underline;
}
.button-container {
display: flex;
justify-content: space-between;
margin-top: 20px;
padding-top: 20px;
border-top: 1px solid #ccc;
}
.action-button {
background-color: #ff7e43;
color: white;
border: none;
padding: 10px 20px;
border-radius: 5px;
cursor: pointer;
}
.action-button:hover {
background-color: #ff621a;
}
.cancel-button {
background-color: #6c757d;
}
.cancel-button:hover {
background-color: #5a6268;
}
`);
// 添加开始按钮
const startButton = document.createElement('button');
startButton.textContent = '点击收集';
startButton.classList.add('start-button');
document.body.appendChild(startButton);
// 点击开始按钮触发事件
startButton.addEventListener('click', async () => {
// 显示正在收集信息
showMessagePopup('正在收集');
// 自动滚动获取ajax数据
const mediaUrls = await scrollAndCollectMedia(targetIds);
// 关闭正在收集弹窗
closeMessagePopup();
// 显示收集完毕信息
showMessagePopup('收集完毕,请选择你需要的照片或视频');
setTimeout(() => {
closeMessagePopup();
// 显示选择弹窗
showSelectionPopup(mediaUrls);
}, 1000);
});
// 自动滚动并收集媒体链接
async function scrollAndCollectMedia(targetIds) {
// 记录当前滚动位置
const originalScrollTop = window.scrollY;
// 模拟鼠标悬停触发视频加载
await triggerVideoLoad();
// 较慢滚动到页面底部
await slowScrollToBottom();
// 滚动回顶部
window.scrollTo({
top: 0, // 滚动到页面顶部
behavior: 'smooth'
});
await new Promise(resolve => setTimeout(resolve, 20)); // 等待滚动完成
// 等待 AJAX 内容加载完成
await waitForAjaxContent();
const mediaUrls = [];
for (const id of targetIds) {
const container = document.getElementById(id);
const classMediaUrls = [];
if (container) {
// 查找图片和视频
const images = container.querySelectorAll('img');
const videos = container.querySelectorAll('video');
for (const img of images) {
const src = img.src;
if (src && !mediaUrls.includes(src) &&!shouldFilter(img)) {
mediaUrls.push(src);
}
}
videos.forEach(video => {
const src = video.src;
if (src && !mediaUrls.includes(src)) {
mediaUrls.push(src);
}
});
}
if (mediaUrls.length > 0) {
console.log(`从 id 为 ${id} 的元素中获取到的图片 URL:`, mediaUrls);
}
}
// 查找 class 为 innerWrapClass 下的图片
if (innerWrapClass) {
const innerWrapImages = document.querySelectorAll(`.${innerWrapClass} img`);
const innerWrapMediaUrls = [];
if (innerWrapImages.length === 0) {
console.log(`未找到 class 为 ${innerWrapClass} 的图片元素,跳过`);
} else {
innerWrapImages.forEach(img => {
let src = img.src;
// 暂时禁用过滤函数
// const filtered = shouldFilter(img);
// console.log(`是否被过滤: ${filtered}`);
if (src.endsWith('_.webp')) {
src = src.replace('_.webp', '');
}
if (src && !mediaUrls.includes(src)) {
mediaUrls.push(src);
innerWrapMediaUrls.push(src);
}
});
if (innerWrapMediaUrls.length > 0) {
console.log(`从 class 为 ${innerWrapClass} 的元素中获取到的图片 URL:`, innerWrapMediaUrls);
}
}
}
// 查找 videoxContainerClass 里的视频
if (videoxContainerClass) {
const videoxVideos = document.querySelectorAll(`.${videoxContainerClass} video`);
const VideosMediaUrls = [];
if (videoxVideos.length === 0) {
console.log(`未找到 class 为 ${videoxContainerClass} 的视频元素,跳过`);
} else {
videoxVideos.forEach(video => {
const src = video.src;
if (src && !mediaUrls.includes(src)) {
mediaUrls.push(src);
VideosMediaUrls.push(src);
}
});
if (VideosMediaUrls.length > 0) {
console.log(`从 class 为 ${videoxContainerClass} 的元素中获取到的图片 URL:`, VideosMediaUrls);
}
}
}
// 滚动回原来的位置
window.scrollTo({
top: originalScrollTop,
behavior: 'smooth'
});
console.log('收集到的媒体链接:', mediaUrls);
return mediaUrls;
}
// 较慢滚动到页面底部
async function slowScrollToBottom() {
const initialScrollStep = 300; // 初始滚动步长
const scrollInterval = 10; // 滚动间隔时间(毫秒)
let scrollStep = initialScrollStep;
let maxScrollTop = document.body.scrollHeight - window.innerHeight;
let currentScrollTop = window.scrollY;
while (currentScrollTop < maxScrollTop) {
window.scrollBy(0, scrollStep);
await new Promise(resolve => setTimeout(resolve, scrollInterval));
currentScrollTop = window.scrollY;
maxScrollTop = document.body.scrollHeight - window.innerHeight;
// 如果接近底部,减小滚动步长
if (maxScrollTop - currentScrollTop < scrollStep) {
scrollStep = maxScrollTop - currentScrollTop;
}
}
// 等待一段时间,检查页面高度是否变化
await new Promise(resolve => setTimeout(resolve, 20));
maxScrollTop = document.body.scrollHeight - window.innerHeight;
currentScrollTop = window.scrollY;
// 如果页面高度变化,继续滚动
while (currentScrollTop < maxScrollTop) {
window.scrollBy(0, scrollStep);
await new Promise(resolve => setTimeout(resolve, scrollInterval));
currentScrollTop = window.scrollY;
maxScrollTop = document.body.scrollHeight - window.innerHeight;
// 如果接近底部,减小滚动步长
if (maxScrollTop - currentScrollTop < scrollStep) {
scrollStep = maxScrollTop - currentScrollTop;
}
}
}
// 模拟鼠标悬停触发视频加载
async function triggerVideoLoad() {
if (thumbnailsClass && thumbnailClass) {
const ulElement = document.querySelector(`.${thumbnailsClass}`);
if (!ulElement) {
console.error(`未找到 .${thumbnailsClass} 元素`);
return;
}
const firstLi = ulElement.querySelector(`.${thumbnailClass}`);
if (!firstLi) {
console.error(`未找到 .${thumbnailClass} 元素`);
return;
}
// 使用 MouseEvent 模拟鼠标悬停
const mouseOverEvent = new MouseEvent('mouseover', {
bubbles: true,
cancelable: true
});
firstLi.dispatchEvent(mouseOverEvent);
// 使用 MutationObserver 检测视频元素加载
await new Promise(resolve => {
const observer = new MutationObserver((mutationsList) => {
for (const mutation of mutationsList) {
if (mutation.type === 'childList') {
const videos = document.querySelectorAll('video');
if (videos.length > 0) {
observer.disconnect();
resolve();
}
}
}
});
observer.observe(document.body, { childList: true, subtree: true });
// 设置超时时间,避免无限等待
setTimeout(() => {
observer.disconnect();
resolve();
}, 500);
});
}
}
// 等待 AJAX 内容加载完成
function waitForAjaxContent() {
return new Promise((resolve) => {
const observer = new MutationObserver((mutationsList) => {
for (const mutation of mutationsList) {
if (mutation.type === 'childList') {
// 有子节点变化,说明可能有新内容加载
observer.disconnect();
// 再等待一段时间,确保内容完全加载
setTimeout(() => {
resolve();
}, 1000);
break;
}
}
});
observer.observe(document.body, { childList: true, subtree: true });
// 设置一个较长的超时时间,避免无限等待
setTimeout(() => {
observer.disconnect();
resolve();
}, 4000);
});
}
// 显示消息弹窗
function showMessagePopup(message) {
const popup = document.createElement('div');
popup.classList.add('message-popup');
popup.textContent = message;
document.body.appendChild(popup);
}
// 关闭消息弹窗
function closeMessagePopup() {
const popup = document.querySelector('.message-popup');
if (popup) {
document.body.removeChild(popup);
}
}
// 显示选择弹窗
function showSelectionPopup(mediaUrls) {
const popup = document.createElement('div');
popup.classList.add('selection-popup');
const title = document.createElement('h2');
title.textContent = '请点击选择需要的照片或视频进行打包';
popup.appendChild(title);
const grid = document.createElement('div');
grid.classList.add('grid');
const selectedUrls = [];
mediaUrls.forEach(url => {
const item = document.createElement('div');
item.classList.add('media-item');
const img = document.createElement('img');
img.src = url;
img.addEventListener('click', () => {
if (selectedUrls.includes(url)) {
selectedUrls.splice(selectedUrls.indexOf(url), 1);
img.classList.remove('selected');
} else {
selectedUrls.push(url);
img.classList.add('selected');
}
});
const checkmark = document.createElement('span');
checkmark.classList.add('checkmark');
checkmark.textContent = '√';
const urlText = document.createElement('p');
const urlLink = document.createElement('a');
urlLink.href = url;
urlLink.target = '_blank';
urlLink.textContent = url;
urlText.appendChild(urlLink);
item.appendChild(img);
item.appendChild(checkmark);
item.appendChild(urlText);
grid.appendChild(item);
});
const cancelButton = document.createElement('button');
cancelButton.textContent = '取消';
cancelButton.classList.add('action-button', 'cancel-button');
cancelButton.addEventListener('click', () => {
document.body.removeChild(popup);
});
const downloadButton = document.createElement('button');
downloadButton.textContent = '一键打包';
downloadButton.classList.add('action-button');
const developerNameElement = document.createElement('span');
developerNameElement.textContent = 'By:AooMing';
developerNameElement.style.margin = '10px 10px';
developerNameElement.style.opacity = '.2';
downloadButton.addEventListener('click', async () => {
if (selectedUrls.length > 0) {
try {
// 显示正在打包的消息弹窗
showMessagePopup('正在打包,请稍候...');
console.log('开始打包,选择的文件数量:', selectedUrls.length);
await downloadSelectedFiles(selectedUrls);
console.log('打包完成,开始下载zip文件');
// 关闭正在打包的消息弹窗
closeMessagePopup();
document.body.removeChild(popup);
} catch (error) {
// 关闭正在打包的消息弹窗
closeMessagePopup();
console.error('打包过程中出现错误:', error);
alert('打包过程中出现错误,请查看控制台日志。');
}
}
});
const buttonContainer = document.createElement('div');
buttonContainer.classList.add('button-container');
buttonContainer.appendChild(cancelButton);
buttonContainer.appendChild(developerNameElement);
buttonContainer.appendChild(downloadButton);
popup.appendChild(grid);
popup.appendChild(buttonContainer);
document.body.appendChild(popup);
}
// 下载选择的文件并打包成zip
async function downloadSelectedFiles(selectedUrls) {
const title = document.title.replace(/[\/:*?"<>|]/g, '_'); // 去除文件名中的非法字符
const batchSize = 10; // 每批处理的文件数量
const validFormats = ['jpg', 'jpeg', 'png', 'mp4', 'avi' , 'webp']; // 支持的文件格式
const zip = new PizZip();
console.log('开始下载选中的文件并添加到zip包中');
for (let i = 0; i < selectedUrls.length; i += batchSize) {
const batchUrls = selectedUrls.slice(i, i + batchSize);
const batchPromises = batchUrls.map(async (url, index) => {
const ext = url.split('.').pop().split('?')[0].toLowerCase();
const type = ext.match(/(jpg|jpeg|png|gif)/i) ? 'image' : 'video';
const timestamp = new Date().getTime();
const filename = `${type}_${timestamp}.${ext}`;
if (!validFormats.includes(ext)) {
console.error(`不支持的文件格式,跳过:`, url);
return;
}
console.log(`开始下载文件 (${i + index + 1}/${selectedUrls.length}):`, url);
await new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: 'GET',
url: url,
responseType: 'blob',
headers: {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36',
'Accept': 'image/*, video/*'
},
onload: async function (response) {
console.log(`文件下载成功 (${i + index + 1}/${selectedUrls.length}):`, url);
try {
const arrayBuffer = await response.response.arrayBuffer();
zip.file(filename, arrayBuffer);
resolve();
} catch (err) {
console.error(`处理文件时出错 (${i + index + 1}/${selectedUrls.length}):`, url, err);
resolve();
}
},
onerror: function (error) {
console.error(`文件下载失败 (${i + index + 1}/${selectedUrls.length}):`, url, error);
resolve();
},
ontimeout: function () {
console.error(`文件下载超时 (${i + index + 1}/${selectedUrls.length}):`, url);
resolve();
}
});
});
});
await Promise.all(batchPromises);
}
console.log('所有文件下载完成,开始生成zip文件');
try {
const startTime = Date.now();
const zipBlob = zip.generate({ type: 'blob' });
const endTime = Date.now();
console.log(`zip文件生成成功,耗时: ${(endTime - startTime) / 1000} 秒,开始保存`);
saveAs(zipBlob, `${title}.zip`);
console.log('zip文件保存成功');
} catch (error) {
console.error('生成zip文件时出现错误:', error);
throw error;
}
}
// 检查图片是否需要过滤
function shouldFilter(img) {
// 检查图片格式是否为gif
if (img.type === 'image/gif') {
return true;
}
// 检查图片高度是否低于100像素
if (img.height < 100) {
return true;
}
// 如果图片格式不是gif且高度不低于100像素,则不过滤
return false;
}
})();