XHS-Downloader

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

当前为 2024-03-30 提交的版本,查看 最新版本

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