您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
提取小红书作品链接,下载小红书无水印图文/视频作品文件
当前为
// ==UserScript== // @name XHS-Downloader // @namespace https://github.com/JoeanAmier/XHS-Downloader // @version 1.3 // @description 提取小红书作品链接,下载小红书无水印图文/视频作品文件 // @author JoeanAmier // @match http*://www.xiaohongshu.com/explore* // @match http*://www.xiaohongshu.com/user/profile/* // @icon  // @grant GM_getValue // @grant GM_setValue // @grant unsafeWindow // @grant GM_setClipboard // @grant GM_registerMenuCommand // @license GNU General Public License v3.0 // ==/UserScript== (function () { let settings = { novice: GM_getValue("novice", true), scroll: GM_getValue("scroll", true), prompt: GM_getValue("prompt", true), }; const menuCommand = [["提取链接说明", "novice"], ["自动滚动屏幕", "scroll"], ["文件下载提示", "prompt"],]; menuCommand.forEach(([a, b]) => { GM_registerMenuCommand(`${a} ${settings[b] ? '✔️' : '❌'}`, function () { settings[b] = !settings[b]; GM_setValue(b, settings[b]); alert('修改设置成功!'); }); }); const icon = ""; const abnormal = () => { alert("提取无水印作品文件下载地址失败!请及时告知作者修复!\n项目地址:https://github.com/JoeanAmier/XHS-Downloader"); }; const downloadPrompt = () => { alert("即将开始批量下载无水印作品文件,请勿多次点击下载按钮!\n此提示可在 Tampermonkey 菜单永久关闭!\n脚本会在后台处理文件并下载,请耐心等待!"); }; const generateVideoUrl = note => { try { return [`https://sns-video-hw.xhscdn.com/${note.video.consumer.originVideoKey}`]; } catch (error) { console.error("Error generating video URL:", error); return []; } }; const generateImageUrl = note => { let images = note.imageList; const regex = /\/([^\/]+?)!/; let urls = []; try { images.forEach((item) => { let match = item.urlDefault.match(regex); if (match && match[1]) { urls.push(`https://ci.xiaohongshu.com/${match[1]}?imageView2/2/w/format/png`); } }) return urls } catch (error) { console.error("Error generating image URLs:", error); return []; } }; const download = async (urls, type_) => { const name = extractName(); if (settings.prompt) { downloadPrompt(); } if (type_ === "video") { await downloadVideo(urls[0], name); } else { await downloadImage(urls, name); } }; const exploreDeal = async note => { try { let links; if (note.type === "normal") { links = generateImageUrl(note); } else { links = generateVideoUrl(note); } if (links.length > 0) { await download(links, note.type); } else { abnormal() } } catch (error) { console.error("Error in deal function:", error); abnormal(); } }; const extractNoteInfo = () => { let note = Object.values(unsafeWindow.__INITIAL_STATE__.note.noteDetailMap); return note[note.length - 1] }; const extractDownloadLinks = async () => { let note = extractNoteInfo(); if (note.note) { await exploreDeal(note.note); } else { abnormal(); } }; const downloadFile = async (link, filename) => { try { // 使用 fetch 获取文件数据 let response = await fetch(link); let blob = await response.blob(); // 创建 Blob 对象的 URL let blobUrl = window.URL.createObjectURL(blob); // 创建一个临时链接元素 let tempLink = document.createElement('a'); tempLink.href = blobUrl; tempLink.download = filename; // 模拟点击链接 tempLink.click(); // 清理临时链接元素 window.URL.revokeObjectURL(blobUrl); } catch (error) { console.error(`下载失败 (${filename}):`, error); } } const extractName = () => { let name = document.title.replace(/[^\u4e00-\u9fa5a-zA-Z0-9]/g, ""); let match = window.location.href.match(/\/([^\/]+)$/); let id = match ? match[1] : null; return name === "" ? id : name }; const downloadVideo = async (url, name) => { await downloadFile(url, `${name}.mp4`); }; const downloadImage = async (urls, name) => { for (const [index, url] of urls.entries()) { await downloadFile(url, `${name}_${index + 1}.png`); } }; const scrollScreen = (callback, feed = false) => { if (settings.scroll && !feed) { let previousHeight = 0; const scrollInterval = setInterval(() => { const currentHeight = document.body.scrollHeight; if (currentHeight !== previousHeight) { window.scrollTo(0, document.body.scrollHeight); previousHeight = currentHeight; } else { clearInterval(scrollInterval); callback(); } }, 2500); } else { callback(); } }; const extractNotesInfo = order => { const notesRawValue = unsafeWindow.__INITIAL_STATE__.user.notes._rawValue[order]; return new Set(notesRawValue.map(({id}) => id)); }; const extractFeedInfo = () => { const notesRawValue = unsafeWindow.__INITIAL_STATE__.feed.feeds._rawValue; return new Set(notesRawValue.map(({id}) => id)); }; const generateUrls = ids => [...ids].map(id => `https://www.xiaohongshu.com/explore/${id}`).join(" "); const confirmBox = () => confirm("即将开始自动提取当前页面作品链接\n提取完毕会自动将作品链接复制到剪贴板\n脚本将会自动滚动屏幕以便加载更多作品\n亦可手动滚动屏幕后再点击按钮提取链接\n此提示可在 Tampermonkey 菜单永久关闭\n是否立即开始提取?"); const extractAllLinks = (callback, order) => { if (!settings.novice || confirmBox()) { scrollScreen(() => { let ids; if (order >= 0 && order <= 2) { ids = extractNotesInfo(order); } else if (order === -1) { ids = extractFeedInfo() } else { ids = []; } let urlsString = generateUrls(ids); callback(urlsString); }, order === -1) } }; const extractAllLinksEvent = (order = 0) => { extractAllLinks(urlsString => { if (urlsString) { GM_setClipboard(urlsString, "text", () => { alert('作品链接已复制到剪贴板!\n搭配 XHS-Downloader 程序可以实现批量下载作品文件!'); }); } else { alert("未提取到任何作品链接,本脚本仅实现可见即可得,并非破解工具!") } }, order); }; const createContainer = () => { let container = document.createElement('div'); container.id = 'xhsFunctionContainer'; let imgTextContainer = document.createElement('div'); imgTextContainer.id = 'xhsImgTextContainer'; let img = new Image(48, 48); // 确保 icon 变量已定义 img.src = icon; img.style.borderRadius = '50%'; img.style.objectFit = 'cover'; let textDiv = document.createElement('div'); textDiv.id = 'xhsImgTextContainer__text' textDiv.textContent = 'XHS-Downloader'; imgTextContainer.appendChild(img); imgTextContainer.appendChild(textDiv); container.appendChild(imgTextContainer); document.body.appendChild(container); return container; }; const createButton = (id, text, onClick, ...args) => { let button = document.createElement('button'); button.id = id; button.textContent = text; button.addEventListener('click', () => onClick(...args)); return button; }; const updateContainer = buttons => { let container = document.getElementById('xhsFunctionContainer'); if (!container) { container = createContainer(); } // 移除除了 imgTextContainer 以外的所有子元素 Array.from(container.children).forEach(child => { if (child.id !== 'xhsImgTextContainer') { child.remove(); } }); // 添加有效按钮 buttons.forEach(button => { container.appendChild(button); }); }; const buttons = [createButton("Download", "下载无水印作品文件", extractDownloadLinks), createButton("Post", "提取发布作品链接", extractAllLinksEvent, 0), createButton("Collection", "提取收藏作品链接", extractAllLinksEvent, 1), createButton("Favorite", "提取点赞作品链接", extractAllLinksEvent, 2), createButton("Feed", "提取发现作品链接", extractAllLinksEvent, -1),] const run = url => { if (url === "https://www.xiaohongshu.com/explore") { updateContainer(buttons.slice(-1)); } else if (url.includes("https://www.xiaohongshu.com/explore/")) { updateContainer(buttons.slice(0, 1)); } else if (url.includes("https://www.xiaohongshu.com/user/profile/")) { updateContainer(buttons.slice(1, 4)); } }; let currentUrl = window.location.href; // 初始化容器 run(currentUrl) // 设置 MutationObserver 来监听 URL 变化 let observer = new MutationObserver(function () { if (currentUrl !== window.location.href) { currentUrl = window.location.href; run(currentUrl); } }); const config = {childList: true, subtree: true}; observer.observe(document.body, config); const buttonStyle = ` #xhsFunctionContainer { position: fixed; bottom: 15%; background-color: #fff; color: #2f3542; padding: 5px 10px; border-radius: 0 32px 32px 0; box-shadow: 0 3.2px 12px #00000014, 0 5px 24px #0000000a; transition: width 0.25s ease-in-out, border-radius 0.25s ease-in-out, height 0.25s ease-in-out; overflow: hidden; white-space: nowrap; width: 65px; /* 初始宽度 */ height: 60px; text-align: center; font-size: 16px; display: flex; flex-direction: column-reverse; z-index: 99999; } #xhsFunctionContainer:hover { padding: 10px 10px 5px 10px; width: 210px; /* hover时的宽度 */ height: auto; } #xhsFunctionContainer button { cursor: pointer; height: 48px; color: #ff4757; font-size: 14px; font-weight: 600; border-radius: 32px; margin-bottom: 14px; border: 3px #ff4757 solid; } #xhsFunctionContainer button:active { background-color: #ff4757; /* 点击时的背景颜色 */ } #xhsImgTextContainer { display: flex; align-items: center; gap: 14px; } #xhsImgTextContainer__text { font-size: 14px; font-weight: 600; } `; const head = document.head || document.getElementsByTagName('head')[0]; const style = document.createElement('style'); head.appendChild(style); style.type = 'text/css'; style.appendChild(document.createTextNode(buttonStyle)); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址