您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
支持微博视频、图片保存
// ==UserScript== // @name 微博视频保存 // @namespace https://yffjglcms.com/ // @version 20241007 // @description 支持微博视频、图片保存 // @author yffjglcms // @match *://s.weibo.com/* // @icon  // @grant none // @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.min.js // @license MIT // ==/UserScript== (function() { 'use strict'; // Your code here... // card-act // thumbnail // 如果存在则执行 function ifExist(selector, func, timeout){ console.log('check ifExist ...' + selector) if(!timeout){ timeout = 5000 } if(document.querySelector(selector)==null) { setTimeout( function(){ifExist(selector, func);}, timeout); } else { func() } } function closest(selector, element) { // 检查当前元素是否匹使用选择器 if (element.matches(selector)) { return element; } // 如果当前元素不匹配,则查找父元素 var parent = element.parentNode; // 如果没有父元素,则已到达文档的顶部,返回null if (!parent) { return null; } // 在父元素的子节点中查找匹配的元素 var closestElement = parent.querySelector(selector); // 如果找到了,则返回该元素 if (closestElement) { return closestElement; } // 否则递归调用closest函数 return closest(selector, parent); } function renderDownload(){ // $('.card .card-act ul').append(`<li><a title="">下载</a></li>`) var ulElements = document.querySelectorAll('.card .card-act ul'); console.log("uls:::", ulElements) ulElements.forEach(function(ul) { var cardFeed = closest('.card-feed', ul); if (!cardFeed) return var newLi = document.createElement('li'); var newA = document.createElement('a'); newA.title = ''; newA.textContent = '下载视频'; // 在这里,cardFeed就是<ul>元素最近的具有'card-feed'类的祖先元素 // console.log(cardFeed); var cardVideoSrc = cardFeed.querySelector('video')?.src if(cardVideoSrc){ // console.log(cardVideoSrc); newA.href=cardVideoSrc newLi.appendChild(newA); ul.appendChild(newLi); } }) } function renderDownloadImg(){ // $('.card .card-act ul').append(`<li><a title="">下载</a></li>`) var ulElements = document.querySelectorAll('.card .card-act ul'); console.log("uls:::", ulElements) ulElements.forEach(function(ul) { var cardFeed = closest('.card-feed', ul); if (!cardFeed) return var newLi = document.createElement('li'); var newA = document.createElement('a'); newA.title = ''; newA.textContent = '下载图片'; // 在这里,cardFeed就是<ul>元素最近的具有'card-feed'类的祖先元素 // console.log(cardFeed); var imgSrc = [...cardFeed.querySelectorAll('img')].filter(x=>x.src.indexOf('thumb150') > 0).map(x => x.src.replaceAll('thumb150', 'large')); if(imgSrc && imgSrc.length > 0){ console.log("imgSrc:::", imgSrc); newA.href='javascript:void(0);' newLi.appendChild(newA); newA.addEventListener('click', async () => { const imageUrls = imgSrc; console.log("imageUrls:::", imageUrls); const zip = new JSZip(); const promises = imageUrls.map(async (url, index) => { const resp = await fetch(url); if (!resp.ok) { throw new Error(`Failed to fetch ${url}`); } const blob = await resp.blob(); zip.file(`image${index + 1}.${url.split('.').pop()}`, blob); }); try { await Promise.all(promises); const content = await zip.generateAsync({ type: 'blob' }); saveAs(content, 'images.zip'); } catch (error) { console.error('Error downloading images or creating ZIP:', error); } }); ul.appendChild(newLi); } }); } ifExist(".thumbnail", renderDownload, 100) ifExist(".media", renderDownloadImg, 100) })();