您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
提取小红书作品链接,下载小红书无水印图文/视频作品文件
当前为
// ==UserScript== // @name XHS-Downloader // @namespace https://github.com/JoeanAmier/XHS-Downloader // @version 1.2 // @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) }; const menuCommand = [["二次确认", "novice"], ["自动滚动", "scroll"]]; menuCommand.forEach(([a, b]) => { GM_registerMenuCommand(`${a} ${settings[b] ? '✔️' : '❌'}`, function (command) { settings[b] = !settings[b]; GM_setValue(b, settings[b]); alert('修改设置成功!'); }); }); const icon = ""; function exploreDeal(note) { try { let links; if (note.type === "normal") { links = generate_image_url(note); } else { links = generate_video_url(note); } if (links.length > 0) { download(links, note.type); } else { abnormal() } } catch (error) { console.error("Error in deal function:", error); abnormal(); } } function extractDownloadLinks() { let note = extractNoteInfo(); if (note.note) { exploreDeal(note.note); } else { abnormal(); } } function extractNoteInfo() { let note = Object.values(unsafeWindow.__INITIAL_STATE__.note.noteDetailMap); return note[note.length - 1] } function generate_video_url(note) { try { return [`https://sns-video-hw.xhscdn.com/${note.video.consumer.originVideoKey}`]; } catch (error) { console.error("Error generating video URL:", error); return []; } } function generate_image_url(note) { let images = note.imageList; const regex = /http:\/\/sns-webpic-qc\.xhscdn\.com\/\d+?\/\S+?\/(\S+?)!/; let urls = []; try { images.forEach((item) => { let match = item.urlDefault.match(regex); if (match && match[1]) { urls.push(`https://sns-img-bd.xhscdn.com/${match[1]}`); } }) return urls } catch (error) { console.error("Error generating image URLs:", error); return []; } } function abnormal() { alert("提取无水印作品文件下载地址失败!请及时告知作者修复!\n项目地址:https://github.com/JoeanAmier/XHS-Downloader"); } function download(urls, type_) { if (type_ === "video") { download_video(urls[0]); } else { download_image(urls); } } function download_video(url) { const name = extract_name() download_file(url, `${name}.mp4`); } function download_image(urls) { const name = extract_name() if (urls.length > 1) { show_urls(urls, name); } else { urls.forEach(function (url, index) { download_file(url, `${name}_${index}.webp`); }) } } function show_urls(urls, name) { let page = window.open(); page.document.title = 'XHS-Downloader'; let container = page.document.createElement('div'); container.style.textAlign = 'center'; container.style.position = 'absolute'; container.style.top = '10%'; container.style.left = '50%'; container.style.transform = 'translate(-50%, 0%)'; container.style.width = '50%'; container.style.height = '50%'; let styleElement = page.document.createElement('style'); styleElement.textContent = ` .XHS-Downloader { bottom: 15%; left: 5%; padding: 15px; background: rgba(123, 237, 159, 0.5); color: #2f3542; border-radius: 15px; cursor: pointer; margin: 5px; } .XHS-Downloader:hover { background: rgba(46, 213, 115, 0.5); } `; page.document.head.appendChild(styleElement); let imgElement = page.document.createElement('img'); imgElement.src = icon; imgElement.style.width = "64px"; container.appendChild(imgElement); let titleElement = page.document.createElement('h3'); titleElement.textContent = "XHS-Downloader"; container.appendChild(titleElement); page.document.body.appendChild(container); let textElement = page.document.createElement('p'); textElement.textContent = "由于浏览器的安全策略限制,无法自动打开多个下载页面,请手动下载图文作品文件!"; container.appendChild(textElement); textElement = page.document.createElement('p'); textElement.textContent = "图片文件可能是 JPG 或 WEBP 格式;如果是 WEBP 格式,下载的文件会有错误的名称后缀!"; container.appendChild(textElement); textElement = page.document.createElement('p'); textElement.textContent = "手动修改为 webp 后缀即可;未来可能会优化;下载图片格式取决于小红书服务器!"; container.appendChild(textElement); urls.forEach((link, index) => { let linkElement = page.document.createElement('a'); linkElement.href = link; linkElement.target = "_blank"; let buttonElement = page.document.createElement('button'); buttonElement.textContent = `无水印图片-${index + 1}`; buttonElement.className = 'XHS-Downloader'; linkElement.setAttribute("download", `${name}_${index + 1}.webp`); linkElement.appendChild(buttonElement); container.appendChild(linkElement); }); page.document.body.appendChild(container); textElement = page.document.createElement('p'); textElement.textContent = "开源协议:GNU General Public License v3.0"; container.appendChild(textElement); textElement = page.document.createElement('p'); let linkElement = page.document.createElement('a'); textElement.textContent = "项目地址:"; linkElement.href = "https://github.com/JoeanAmier/XHS-Downloader"; linkElement.textContent = "https://github.com/JoeanAmier/XHS-Downloader"; linkElement.target = "_blank"; textElement.appendChild(linkElement); container.appendChild(textElement); let favicon = page.document.createElement('link'); favicon.rel = "icon"; favicon.type = "image/x-icon"; favicon.href = icon; page.document.head.appendChild(favicon); } function extract_name() { 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 } function download_file(url, name) { let file = document.createElement('a'); file.href = url; file.download = name; file.target = "_blank"; document.body.appendChild(file); file.click(); document.body.removeChild(file); } function scrollScreen(callback, feed = false) { if (settings.scroll && !feed) { let previousHeight = 0; const scrollInterval = setInterval(() => { const currentHeight = document.body.scrollHeight; if (currentHeight !== previousHeight) { scrollToBottom(); previousHeight = currentHeight; } else { clearInterval(scrollInterval); callback(); } }, 1500); } else { callback(); } function scrollToBottom() { window.scrollTo(0, document.body.scrollHeight); } } function extractNotesInfo(order) { const notesRawValue = unsafeWindow.__INITIAL_STATE__.user.notes._rawValue[order]; return new Set(notesRawValue.map(({id}) => id)); } function extractFeedInfo() { const notesRawValue = unsafeWindow.__INITIAL_STATE__.feed.feeds._rawValue; return new Set(notesRawValue.map(({id}) => id)); } function generateUrls(ids) { return [...ids].map(id => `https://www.xiaohongshu.com/explore/${id}`).join(" "); } function confirmBox() { return confirm("即将开始自动提取当前页面作品链接\n提取完毕会自动将作品链接复制到剪贴板\n脚本会自动滚动屏幕以便加载更多作品(可关闭)\n此提示可在 Tampermonkey 菜单永久关闭\n是否立即开始提取?"); } function 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) } } function extractAllLinksEvent(order = 0) { extractAllLinks(urlsString => { if (urlsString) { GM_setClipboard(urlsString, "text", () => { alert('作品链接已复制到剪贴板!\n搭配 XHS-Downloader 程序可以实现批量下载作品文件!'); }); } else { alert("未提取到任何作品链接!") } }, order); } function 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; } function createButton(id, text, onClick, ...args) { let button = document.createElement('button'); button.id = id; button.textContent = text; button.addEventListener('click', () => onClick(...args)); return button; } function 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),] function 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 (mutationsList, observer) { 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)); })();