XHS-Downloader

提取小红书作品/用户链接,下载小红书无水印图文/视频作品文件

目前为 2024-08-30 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name XHS-Downloader
  3. // @namespace https://github.com/JoeanAmier/XHS-Downloader
  4. // @version 1.6.2
  5. // @description 提取小红书作品/用户链接,下载小红书无水印图文/视频作品文件
  6. // @author JoeanAmier
  7. // @match http*://xhslink.com/*
  8. // @match http*://www.xiaohongshu.com/explore*
  9. // @match http*://www.xiaohongshu.com/user/profile/*
  10. // @match http*://www.xiaohongshu.com/search_result*
  11. // @match http*://www.xiaohongshu.com/board/*
  12. // @icon 
  13. // @grant GM_getValue
  14. // @grant GM_setValue
  15. // @grant unsafeWindow
  16. // @grant GM_setClipboard
  17. // @grant GM_registerMenuCommand
  18. // @license GNU General Public License v3.0
  19. // ==/UserScript==
  20.  
  21. (function () {
  22. let disclaimer = GM_getValue("disclaimer", null);
  23.  
  24. const readme = () => {
  25. const instructions = `
  26. 关于 XHS-Downloader 用户脚本的功能说明:
  27.  
  28. 功能清单:
  29.  
  30. 1. 下载小红书无水印作品文件
  31. 2. 提取发现页面作品链接
  32. 3. 提取账号发布作品链接
  33. 4. 提取账号收藏作品链接
  34. 5. 提取账号专辑作品链接
  35. 6. 提取账号点赞作品链接
  36. 7. 提取搜索结果作品链接
  37. 8. 提取搜索结果用户链接
  38.  
  39. 详细说明:
  40.  
  41. 1. 下载小红书无水印作品文件时,脚本需要花费时间处理文件,请等待片刻,切勿多次点击下载按钮
  42. 2. 无水印图片文件为 PNG 格式;无水印视频文件较大,可能需要较长的时间处理,页面跳转可能会导致下载失败
  43. 3. 提取账号发布、收藏、点赞、专辑作品链接时,脚本会尝试自动滚动屏幕直至加载全部作品,滚动检测间隔:2.5
  44. 4. 提取发现作品链接、搜索作品、用户链接时,脚本会自动滚动屏幕以尝试加载更多内容,滚动屏幕次数:10
  45. 5. 可以修改滚动检测间隔、滚动屏幕次数,修改后立即生效;亦可关闭自动滚动屏幕功能,手动滚动屏幕加载内容
  46. 6. 使用全局代理工具可能会导致脚本下载文件失败,如有异常,请尝试关闭代理工具,必要时向作者反馈
  47. 7. XHS-Downloader 用户脚本仅实现可见即可得的数据采集功能,无任何收费功能和破解功能
  48.  
  49. 项目开源地址:https://github.com/JoeanAmier/XHS-Downloader
  50. `
  51. const disclaimer_content = `
  52. 关于 XHS-Downloader 免责声明:
  53.  
  54. 1. 使用者对本项目的使用由使用者自行决定,并自行承担风险。作者对使用者使用本项目所产生的任何损失、责任、或风险概不负责。
  55. 2. 本项目的作者提供的代码和功能是基于现有知识和技术的开发成果。作者尽力确保代码的正确性和安全性,但不保证代码完全没有错误或缺陷。
  56. 3. 使用者在使用本项目时必须严格遵守 GNU General Public License v3.0 的要求,并在适当的地方注明使用了 GNU General Public License v3.0 的代码。
  57. 4. 使用者在任何情况下均不得将本项目的作者、贡献者或其他相关方与使用者的使用行为联系起来,或要求其对使用者使用本项目所产生的任何损失或损害负责。
  58. 5. 使用者在使用本项目的代码和功能时,必须自行研究相关法律法规,并确保其使用行为合法合规。任何因违反法律法规而导致的法律责任和风险,均由使用者自行承担。
  59. 6. 本项目的作者不会提供 XHS-Downloader 项目的付费版本,也不会提供与 XHS-Downloader 项目相关的任何商业服务。
  60. 7. 基于本项目进行的任何二次开发、修改或编译的程序与原创作者无关,原创作者不承担与二次开发行为或其结果相关的任何责任,使用者应自行对因二次开发可能带来的各种情况负全部责任。
  61.  
  62. 在使用本项目的代码和功能之前,请您认真考虑并接受以上免责声明。如果您对上述声明有任何疑问或不同意,请不要使用本项目的代码和功能。如果您使用了本项目的代码和功能,则视为您已完全理解并接受上述免责声明,并自愿承担使用本项目的一切风险和后果。
  63.  
  64. 是否已阅读 XHS-Downloader 功能说明与免责声明(YES/NO)
  65. `
  66. alert(instructions);
  67. if (!disclaimer) {
  68. const answer = prompt(disclaimer_content, "");
  69. if (answer === null) {
  70. GM_setValue("disclaimer", false);
  71. disclaimer = false;
  72. } else {
  73. GM_setValue("disclaimer", answer.toUpperCase() === "YES");
  74. disclaimer = GM_getValue("disclaimer");
  75. location.reload();
  76. }
  77. }
  78. };
  79.  
  80. if (disclaimer === null) {
  81. readme();
  82. }
  83.  
  84. GM_registerMenuCommand("关于 XHS-Downloader", function () {
  85. readme();
  86. });
  87.  
  88. let scroll = GM_getValue("scroll", true);
  89.  
  90. GM_registerMenuCommand(`自动滚动屏幕功能 ${scroll ? '✔️' : '❌'}`, function () {
  91. scroll = !scroll;
  92. GM_setValue("scroll", scroll);
  93. alert('修改自动滚动屏幕功能成功!');
  94. });
  95.  
  96. let timeout = GM_getValue("timeout", 2500);
  97.  
  98. GM_registerMenuCommand("修改滚动检测间隔", function () {
  99. let data;
  100. data = prompt("请输入自动滚动屏幕检测间隔:\n如果网络环境不佳导致脚本未能加载全部作品,可以设置较大的检测间隔!", timeout / 1000);
  101. if (data === null) {
  102. return
  103. }
  104. data = parseFloat(data) || 2.5
  105. timeout = data * 1000;
  106. GM_setValue("timeout", timeout);
  107. alert(`修改自动滚动屏幕检测间隔成功,当前值:${data} 秒`);
  108. });
  109.  
  110. let number = GM_getValue("number", 10);
  111.  
  112. GM_registerMenuCommand("修改滚动屏幕次数", function () {
  113. let data;
  114. data = prompt("请输入自动滚动屏幕次数:\n仅对提取发现作品、搜索作品、搜索用户链接生效!", number);
  115. if (data === null) {
  116. return
  117. }
  118. number = parseInt(data) || 10;
  119. GM_setValue("number", number);
  120. alert(`修改自动滚动屏幕次数成功,当前值:${number} 次`);
  121. });
  122.  
  123. const icon = "";
  124.  
  125. const about = () => {
  126. window.open('https://github.com/JoeanAmier/XHS-Downloader', '_blank');
  127. }
  128.  
  129. const abnormal = () => {
  130. alert("下载无水印作品文件失败!请向作者反馈!\n项目地址:https://github.com/JoeanAmier/XHS-Downloader");
  131. };
  132.  
  133. const generateVideoUrl = note => {
  134. try {
  135. return [`https://sns-video-bd.xhscdn.com/${note.video.consumer.originVideoKey}`];
  136. } catch (error) {
  137. console.error("Error generating video URL:", error);
  138. return [];
  139. }
  140. };
  141.  
  142. const generateImageUrl = note => {
  143. let images = note.imageList;
  144. const regex = /http:\/\/sns-webpic-qc\.xhscdn.com\/\d+\/[0-9a-z]+\/(\S+)!/;
  145. let urls = [];
  146. try {
  147. images.forEach((item) => {
  148. let match = item.urlDefault.match(regex);
  149. if (match && match[1]) {
  150. urls.push(`https://ci.xiaohongshu.com/${match[1]}?imageView2/format/png`);
  151. }
  152. })
  153. return urls
  154. } catch (error) {
  155. console.error("Error generating image URLs:", error);
  156. return [];
  157. }
  158. };
  159.  
  160. const download = async (urls, type_) => {
  161. const name = extractName();
  162. console.info(`基础文件名称 ${name}`);
  163. if (type_ === "video") {
  164. await downloadVideo(urls[0], name);
  165. } else {
  166. await downloadImage(urls, name);
  167. }
  168. };
  169.  
  170. const exploreDeal = async note => {
  171. try {
  172. let links;
  173. if (note.type === "normal") {
  174. links = generateImageUrl(note);
  175. } else {
  176. links = generateVideoUrl(note);
  177. }
  178. if (links.length > 0) {
  179. console.info("无水印文件下载链接", links);
  180. await download(links, note.type);
  181. } else {
  182. abnormal()
  183. }
  184. } catch (error) {
  185. console.error("Error in deal function:", error);
  186. abnormal();
  187. }
  188. };
  189.  
  190. const extractNoteInfo = () => {
  191. const regex = /\/explore\/([^?]+)/;
  192. const match = currentUrl.match(regex);
  193. if (match) {
  194. // let note = Object.values(unsafeWindow.__INITIAL_STATE__.note.noteDetailMap);
  195. return unsafeWindow.__INITIAL_STATE__.note.noteDetailMap[match[1]]
  196. } else {
  197. console.error("从链接提取作品 ID 失败", currentUrl,);
  198. }
  199. };
  200.  
  201. const extractDownloadLinks = async () => {
  202. if (currentUrl.includes("https://www.xiaohongshu.com/explore/")) {
  203. let note = extractNoteInfo();
  204. if (note.note) {
  205. await exploreDeal(note.note);
  206. } else {
  207. abnormal();
  208. }
  209. }
  210. };
  211.  
  212. const downloadFile = async (link, filename) => {
  213. try {
  214. // 使用 fetch 获取文件数据
  215. let response = await fetch(link, {
  216. method: "GET",
  217. });
  218.  
  219. // 检查响应状态码
  220. if (!response.ok) {
  221. console.error(`请求失败,状态码: ${response.status}`, response.status);
  222. return false
  223. }
  224.  
  225. let blob = await response.blob();
  226.  
  227. // 创建 Blob 对象的 URL
  228. let blobUrl = window.URL.createObjectURL(blob);
  229.  
  230. // 创建一个临时链接元素
  231. let tempLink = document.createElement('a');
  232. tempLink.href = blobUrl;
  233. tempLink.download = filename;
  234.  
  235. // 模拟点击链接
  236. tempLink.click();
  237.  
  238. // 清理临时链接元素
  239. window.URL.revokeObjectURL(blobUrl);
  240.  
  241. return true
  242. } catch (error) {
  243. console.error(`下载失败 (${filename}):`, error);
  244. return false
  245. }
  246. }
  247.  
  248. const extractName = () => {
  249. let name = document.title.replace(/[^\u4e00-\u9fa5a-zA-Z0-9]/g, "");
  250. let match = currentUrl.match(/\/([^\/]+)$/);
  251. let id = match ? match[1] : null;
  252. return name === "" ? id : name
  253. };
  254.  
  255. const downloadVideo = async (url, name) => {
  256. if (!await downloadFile(url, `${name}.mp4`)) {
  257. abnormal();
  258. }
  259. };
  260.  
  261. const downloadImage = async (urls, name) => {
  262. let result = [];
  263. for (const [index, url] of urls.entries()) {
  264. result.push(await downloadFile(url, `${name}_${index + 1}.png`));
  265. }
  266. if (!result.every(item => item === true)) {
  267. abnormal();
  268. }
  269. };
  270.  
  271. const scrollScreen = (callback, endless = false) => {
  272. if (!scroll) {
  273. callback();
  274. } else if (endless) {
  275. let previousHeight = 0;
  276. const scrollInterval = setInterval(() => {
  277. const currentHeight = document.body.scrollHeight;
  278. if (currentHeight !== previousHeight) {
  279. window.scrollTo(0, document.body.scrollHeight);
  280. previousHeight = currentHeight;
  281. } else {
  282. clearInterval(scrollInterval);
  283. callback();
  284. }
  285. }, timeout);
  286. } else {
  287. let previousHeight = 0;
  288. let scrollCount = 0;
  289. const scrollInterval = setInterval(() => {
  290. const currentHeight = document.body.scrollHeight;
  291. if (currentHeight !== previousHeight && scrollCount < number) {
  292. window.scrollTo(0, document.body.scrollHeight);
  293. previousHeight = currentHeight;
  294. scrollCount++;
  295. } else {
  296. clearInterval(scrollInterval);
  297. callback();
  298. }
  299. }, timeout);
  300. }
  301. };
  302.  
  303. const extractNotesInfo = order => {
  304. const notesRawValue = unsafeWindow.__INITIAL_STATE__.user.notes._rawValue[order];
  305. return notesRawValue.map(item => [item.id, item.xsecToken]);
  306. };
  307.  
  308. const extractBoardInfo = order => {
  309. // 定义正则表达式来匹配 URL 中的 ID
  310. const regex = /\/board\/([a-z0-9]+)\?/;
  311.  
  312. // 使用 exec 方法执行正则表达式
  313. const match = regex.exec(currentUrl);
  314.  
  315. // 检查是否有匹配
  316. if (match) {
  317. // 提取 ID
  318. const id = match[1]; // match[0] 是整个匹配的字符串,match[1] 是第一个括号内的匹配
  319.  
  320. const notesRawValue = unsafeWindow.__INITIAL_STATE__.board.boardFeedsMap._rawValue[id].notes;
  321. return notesRawValue.map(item => [item.noteId, item.xsecToken]);
  322. } else {
  323. console.error("从链接提取专辑 ID 失败", currentUrl,);
  324. return [];
  325. }
  326. };
  327.  
  328. const extractFeedInfo = () => {
  329. const notesRawValue = unsafeWindow.__INITIAL_STATE__.feed.feeds._rawValue;
  330. return notesRawValue.map(item => [item.id, item.xsecToken]);
  331. };
  332.  
  333. const extractSearchNotes = () => {
  334. const notesRawValue = unsafeWindow.__INITIAL_STATE__.search.feeds._rawValue;
  335. return notesRawValue.map(item => [item.id, item.xsecToken]);
  336. }
  337.  
  338. const extractSearchUsers = () => {
  339. const notesRawValue = unsafeWindow.__INITIAL_STATE__.search.userLists._rawValue;
  340. return notesRawValue.map(item => item.id);
  341. }
  342.  
  343. const generateNoteUrls = data => data.map(([id, token]) => `https://www.xiaohongshu.com/explore/${id}?xsec_token=${token}&xsec_source=pc_feed`).join(" ");
  344.  
  345. const generateUserUrls = data => data.map(id => `https://www.xiaohongshu.com/user/profile/${id}`).join(" ");
  346.  
  347. const extractAllLinks = (callback, order) => {
  348. scrollScreen(() => {
  349. let data;
  350. if (order >= 0 && order <= 2) {
  351. data = extractNotesInfo(order);
  352. } else if (order === 3) {
  353. data = extractSearchNotes();
  354. } else if (order === 4) {
  355. data = extractSearchUsers();
  356. } else if (order === -1) {
  357. data = extractFeedInfo()
  358. } else if (order === 5) {
  359. data = extractBoardInfo()
  360. } else {
  361. data = [];
  362. }
  363. let urlsString = order !== 4 ? generateNoteUrls(data) : generateUserUrls(data);
  364. callback(urlsString);
  365. }, [0, 1, 2, 5].includes(order))
  366. };
  367.  
  368. const extractAllLinksEvent = (order = 0) => {
  369. extractAllLinks(urlsString => {
  370. if (urlsString) {
  371. GM_setClipboard(urlsString, "text", () => {
  372. alert('作品/用户链接已复制到剪贴板!');
  373. });
  374. } else {
  375. alert("未提取到任何作品/用户链接!")
  376. }
  377. }, order);
  378. };
  379.  
  380. const createContainer = () => {
  381. let container = document.createElement('div');
  382. container.id = 'xhsFunctionContainer';
  383.  
  384. let imgTextContainer = document.createElement('div');
  385. imgTextContainer.id = 'xhsImgTextContainer';
  386.  
  387. let img = new Image(48, 48); // 确保 icon 变量已定义
  388. img.src = icon;
  389. img.style.borderRadius = '50%';
  390. img.style.objectFit = 'cover';
  391.  
  392. let textDiv = document.createElement('div');
  393. textDiv.id = 'xhsImgTextContainer__text'
  394. textDiv.textContent = 'XHS-Downloader';
  395.  
  396. imgTextContainer.appendChild(img);
  397. imgTextContainer.appendChild(textDiv);
  398.  
  399. container.appendChild(imgTextContainer);
  400.  
  401. document.body.appendChild(container);
  402. return container;
  403. };
  404.  
  405. const createButton = (id, text, onClick, ...args) => {
  406. let button = document.createElement('button');
  407. button.id = id;
  408. button.textContent = text;
  409. button.addEventListener('click', () => onClick(...args));
  410. return button;
  411. };
  412.  
  413. const exclusionButton = ["xhsImgTextContainer", "About"];
  414.  
  415. const updateContainer = buttons => {
  416. let container = document.getElementById('xhsFunctionContainer');
  417. if (!container) {
  418. container = createContainer();
  419. }
  420.  
  421. // 移除除了 imgTextContainer 以外的所有子元素
  422. Array.from(container.children).forEach(child => {
  423. if (!exclusionButton.includes(child.id)) {
  424. child.remove();
  425. }
  426. });
  427.  
  428. // 添加有效按钮
  429. buttons.forEach(button => {
  430. container.appendChild(button);
  431. });
  432. };
  433.  
  434. const buttons = [createButton("Download", "下载无水印作品文件", extractDownloadLinks), createButton("Post", "提取发布作品链接", extractAllLinksEvent, 0), createButton("Collection", "提取收藏作品链接", extractAllLinksEvent, 1), createButton("Favorite", "提取点赞作品链接", extractAllLinksEvent, 2), createButton("Feed", "提取发现作品链接", extractAllLinksEvent, -1), createButton("Search", "提取搜索作品链接", extractAllLinksEvent, 3), createButton("User", "提取搜索用户链接", extractAllLinksEvent, 4), createButton("Board", "提取专辑作品链接", extractAllLinksEvent, 5), createButton("Disclaimer", "脚本说明及免责声明", readme,), createButton("About", "关于 XHS-Downloader", about,),];
  435.  
  436. const run = url => {
  437. setTimeout(function () {
  438. if (!disclaimer) {
  439. } else if (url === "https://www.xiaohongshu.com/explore" || url.includes("https://www.xiaohongshu.com/explore?")) {
  440. updateContainer(buttons.slice(4, 5));
  441. } else if (url.includes("https://www.xiaohongshu.com/explore/")) {
  442. updateContainer(buttons.slice(0, 1));
  443. } else if (url.includes("https://www.xiaohongshu.com/user/profile/")) {
  444. updateContainer(buttons.slice(1, 4));
  445. } else if (url.includes("https://www.xiaohongshu.com/search_result")) {
  446. updateContainer(buttons.slice(5, 7));
  447. } else if (url.includes("https://www.xiaohongshu.com/board/")) {
  448. updateContainer(buttons.slice(7, 8));
  449. }
  450. }, 500)
  451. }
  452.  
  453. let currentUrl = window.location.href;
  454.  
  455. updateContainer(buttons.slice(8));
  456.  
  457. // 初始化容器
  458. run(currentUrl)
  459.  
  460. // 设置 MutationObserver 来监听 URL 变化
  461. let observer
  462. if (disclaimer) {
  463. observer = new MutationObserver(function () {
  464. if (currentUrl !== window.location.href) {
  465. currentUrl = window.location.href;
  466. run(currentUrl);
  467. }
  468. });
  469. const config = {childList: true, subtree: true};
  470. observer.observe(document.body, config);
  471. }
  472.  
  473. const buttonStyle = `
  474. #xhsFunctionContainer {
  475. position: fixed;
  476. bottom: 15%;
  477. background-color: #fff;
  478. color: #2f3542;
  479. padding: 5px 10px;
  480. border-radius: 0 32px 32px 0;
  481. box-shadow: 0 3.2px 12px #00000014, 0 5px 24px #0000000a;
  482. transition: width 0.25s ease-in-out, border-radius 0.25s ease-in-out, height 0.25s ease-in-out;
  483. overflow: hidden;
  484. white-space: nowrap;
  485. width: 65px; /* 初始宽度 */
  486. height: 60px;
  487. text-align: center;
  488. font-size: 16px;
  489. display: flex;
  490. flex-direction: column-reverse;
  491. z-index: 99999;
  492. }
  493. #xhsFunctionContainer:hover {
  494. padding: 10px 10px 5px 10px;
  495. width: 210px; /* hover时的宽度 */
  496. height: auto;
  497. }
  498.  
  499. #xhsFunctionContainer button {
  500. cursor: pointer;
  501. height: 48px;
  502. color: #ff4757;
  503. font-size: 14px;
  504. font-weight: 600;
  505. border-radius: 32px;
  506. margin-bottom: 14px;
  507. border: 3px #ff4757 solid;
  508. }
  509. #xhsFunctionContainer button:active {
  510. background-color: #ff4757; /* 点击时的背景颜色 */
  511. }
  512. #xhsImgTextContainer {
  513. display: flex;
  514. align-items: center;
  515. gap: 14px;
  516. }
  517. #xhsImgTextContainer__text {
  518. font-size: 14px;
  519. font-weight: 600;
  520. }
  521. `;
  522.  
  523. const head = document.head || document.getElementsByTagName('head')[0];
  524. const style = document.createElement('style');
  525. head.appendChild(style);
  526.  
  527. style.type = 'text/css';
  528. style.appendChild(document.createTextNode(buttonStyle));
  529. console.info("用户接受 XHS-Downloader 免责声明", disclaimer)
  530. })();