XHS-Downloader

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

目前为 2024-01-04 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name XHS-Downloader
  3. // @namespace https://github.com/JoeanAmier/XHS-Downloader
  4. // @version 1.2
  5. // @description 提取小红书作品链接,下载小红书无水印图文/视频作品文件
  6. // @author JoeanAmier
  7. // @match http*://www.xiaohongshu.com/explore*
  8. // @match http*://www.xiaohongshu.com/user/profile/*
  9. // @icon 
  10. // @grant GM_getValue
  11. // @grant GM_setValue
  12. // @grant unsafeWindow
  13. // @grant GM_setClipboard
  14. // @grant GM_registerMenuCommand
  15. // @license GNU General Public License v3.0
  16. // ==/UserScript==
  17.  
  18. (function () {
  19. let settings = {
  20. novice: GM_getValue("novice", true), scroll: GM_getValue("scroll", true)
  21. };
  22.  
  23. const menuCommand = [["二次确认", "novice"], ["自动滚动", "scroll"]];
  24.  
  25. menuCommand.forEach(([a, b]) => {
  26. GM_registerMenuCommand(`${a} ${settings[b] ? '✔️' : '❌'}`, function (command) {
  27. settings[b] = !settings[b];
  28. GM_setValue(b, settings[b]);
  29. alert('修改设置成功!');
  30. });
  31. });
  32.  
  33. const icon = "";
  34.  
  35. function exploreDeal(note) {
  36. try {
  37. let links;
  38. if (note.type === "normal") {
  39. links = generate_image_url(note);
  40. } else {
  41. links = generate_video_url(note);
  42. }
  43. if (links.length > 0) {
  44. download(links, note.type);
  45. } else {
  46. abnormal()
  47. }
  48. } catch (error) {
  49. console.error("Error in deal function:", error);
  50. abnormal();
  51. }
  52. }
  53.  
  54. function extractDownloadLinks() {
  55. let note = extractNoteInfo();
  56. if (note.note) {
  57. exploreDeal(note.note);
  58. } else {
  59. abnormal();
  60. }
  61. }
  62.  
  63. function extractNoteInfo() {
  64. let note = Object.values(unsafeWindow.__INITIAL_STATE__.note.noteDetailMap);
  65. return note[note.length - 1]
  66. }
  67.  
  68. function generate_video_url(note) {
  69. try {
  70. return [`https://sns-video-hw.xhscdn.com/${note.video.consumer.originVideoKey}`];
  71. } catch (error) {
  72. console.error("Error generating video URL:", error);
  73. return [];
  74. }
  75. }
  76.  
  77. function generate_image_url(note) {
  78. let images = note.imageList;
  79. const regex = /http:\/\/sns-webpic-qc\.xhscdn\.com\/\d+?\/\S+?\/(\S+?)!/;
  80. let urls = [];
  81. try {
  82. images.forEach((item) => {
  83. let match = item.urlDefault.match(regex);
  84. if (match && match[1]) {
  85. urls.push(`https://sns-img-bd.xhscdn.com/${match[1]}`);
  86. }
  87. })
  88. return urls
  89. } catch (error) {
  90. console.error("Error generating image URLs:", error);
  91. return [];
  92. }
  93. }
  94.  
  95. function abnormal() {
  96. alert("提取无水印作品文件下载地址失败!请及时告知作者修复!\n项目地址:https://github.com/JoeanAmier/XHS-Downloader");
  97. }
  98.  
  99. function download(urls, type_) {
  100. if (type_ === "video") {
  101. download_video(urls[0]);
  102. } else {
  103. download_image(urls);
  104. }
  105. }
  106.  
  107. function download_video(url) {
  108. const name = extract_name()
  109. download_file(url, `${name}.mp4`);
  110. }
  111.  
  112. function download_image(urls) {
  113. const name = extract_name()
  114. if (urls.length > 1) {
  115. show_urls(urls, name);
  116. } else {
  117. urls.forEach(function (url, index) {
  118. download_file(url, `${name}_${index}.webp`);
  119. })
  120. }
  121. }
  122.  
  123. function show_urls(urls, name) {
  124. let page = window.open();
  125. page.document.title = 'XHS-Downloader';
  126. let container = page.document.createElement('div');
  127. container.style.textAlign = 'center';
  128. container.style.position = 'absolute';
  129. container.style.top = '10%';
  130. container.style.left = '50%';
  131. container.style.transform = 'translate(-50%, 0%)';
  132. container.style.width = '50%';
  133. container.style.height = '50%';
  134.  
  135. let styleElement = page.document.createElement('style');
  136. styleElement.textContent = `
  137. .XHS-Downloader {
  138. bottom: 15%;
  139. left: 5%;
  140. padding: 15px;
  141. background: rgba(123, 237, 159, 0.5);
  142. color: #2f3542;
  143. border-radius: 15px;
  144. cursor: pointer;
  145. margin: 5px;
  146. }
  147.  
  148. .XHS-Downloader:hover {
  149. background: rgba(46, 213, 115, 0.5);
  150. }
  151. `;
  152. page.document.head.appendChild(styleElement);
  153.  
  154. let imgElement = page.document.createElement('img');
  155. imgElement.src = icon;
  156. imgElement.style.width = "64px";
  157. container.appendChild(imgElement);
  158.  
  159. let titleElement = page.document.createElement('h3');
  160. titleElement.textContent = "XHS-Downloader";
  161. container.appendChild(titleElement);
  162.  
  163. page.document.body.appendChild(container);
  164.  
  165. let textElement = page.document.createElement('p');
  166. textElement.textContent = "由于浏览器的安全策略限制,无法自动打开多个下载页面,请手动下载图文作品文件!";
  167. container.appendChild(textElement);
  168.  
  169. textElement = page.document.createElement('p');
  170. textElement.textContent = "图片文件可能是 JPG 或 WEBP 格式;如果是 WEBP 格式,下载的文件会有错误的名称后缀!";
  171. container.appendChild(textElement);
  172.  
  173. textElement = page.document.createElement('p');
  174. textElement.textContent = "手动修改为 webp 后缀即可;未来可能会优化;下载图片格式取决于小红书服务器!";
  175. container.appendChild(textElement);
  176.  
  177. urls.forEach((link, index) => {
  178. let linkElement = page.document.createElement('a');
  179. linkElement.href = link;
  180. linkElement.target = "_blank";
  181.  
  182. let buttonElement = page.document.createElement('button');
  183. buttonElement.textContent = `无水印图片-${index + 1}`;
  184. buttonElement.className = 'XHS-Downloader';
  185.  
  186. linkElement.setAttribute("download", `${name}_${index + 1}.webp`);
  187. linkElement.appendChild(buttonElement);
  188. container.appendChild(linkElement);
  189. });
  190.  
  191. page.document.body.appendChild(container);
  192.  
  193. textElement = page.document.createElement('p');
  194. textElement.textContent = "开源协议:GNU General Public License v3.0";
  195. container.appendChild(textElement);
  196.  
  197. textElement = page.document.createElement('p');
  198. let linkElement = page.document.createElement('a');
  199.  
  200. textElement.textContent = "项目地址:";
  201. linkElement.href = "https://github.com/JoeanAmier/XHS-Downloader";
  202. linkElement.textContent = "https://github.com/JoeanAmier/XHS-Downloader";
  203. linkElement.target = "_blank";
  204.  
  205. textElement.appendChild(linkElement);
  206. container.appendChild(textElement);
  207.  
  208. let favicon = page.document.createElement('link');
  209. favicon.rel = "icon";
  210. favicon.type = "image/x-icon";
  211. favicon.href = icon;
  212. page.document.head.appendChild(favicon);
  213. }
  214.  
  215. function extract_name() {
  216. let name = document.title.replace(/[^\u4e00-\u9fa5a-zA-Z0-9]/g, "");
  217. let match = window.location.href.match(/\/([^\/]+)$/);
  218. let id = match ? match[1] : null;
  219. return name === "" ? id : name
  220. }
  221.  
  222. function download_file(url, name) {
  223. let file = document.createElement('a');
  224. file.href = url;
  225. file.download = name;
  226. file.target = "_blank";
  227. document.body.appendChild(file);
  228. file.click();
  229. document.body.removeChild(file);
  230. }
  231.  
  232. function scrollScreen(callback, feed = false) {
  233. if (settings.scroll && !feed) {
  234. let previousHeight = 0;
  235. const scrollInterval = setInterval(() => {
  236. const currentHeight = document.body.scrollHeight;
  237. if (currentHeight !== previousHeight) {
  238. scrollToBottom();
  239. previousHeight = currentHeight;
  240. } else {
  241. clearInterval(scrollInterval);
  242. callback();
  243. }
  244. }, 1500);
  245. } else {
  246. callback();
  247. }
  248.  
  249. function scrollToBottom() {
  250. window.scrollTo(0, document.body.scrollHeight);
  251. }
  252. }
  253.  
  254. function extractNotesInfo(order) {
  255. const notesRawValue = unsafeWindow.__INITIAL_STATE__.user.notes._rawValue[order];
  256. return new Set(notesRawValue.map(({id}) => id));
  257. }
  258.  
  259. function extractFeedInfo() {
  260. const notesRawValue = unsafeWindow.__INITIAL_STATE__.feed.feeds._rawValue;
  261. return new Set(notesRawValue.map(({id}) => id));
  262. }
  263.  
  264. function generateUrls(ids) {
  265. return [...ids].map(id => `https://www.xiaohongshu.com/explore/${id}`).join(" ");
  266. }
  267.  
  268. function confirmBox() {
  269. return confirm("即将开始自动提取当前页面作品链接\n提取完毕会自动将作品链接复制到剪贴板\n脚本会自动滚动屏幕以便加载更多作品(可关闭)\n此提示可在 Tampermonkey 菜单永久关闭\n是否立即开始提取?");
  270. }
  271.  
  272. function extractAllLinks(callback, order) {
  273. if (!settings.novice || confirmBox()) {
  274. scrollScreen(() => {
  275. let ids;
  276. if (order >= 0 && order <= 2) {
  277. ids = extractNotesInfo(order);
  278. } else if (order === -1) {
  279. ids = extractFeedInfo()
  280. } else {
  281. ids = [];
  282. }
  283. let urlsString = generateUrls(ids);
  284. callback(urlsString);
  285. }, order === -1)
  286. }
  287. }
  288.  
  289. function extractAllLinksEvent(order = 0) {
  290. extractAllLinks(urlsString => {
  291. if (urlsString) {
  292. GM_setClipboard(urlsString, "text", () => {
  293. alert('作品链接已复制到剪贴板!\n搭配 XHS-Downloader 程序可以实现批量下载作品文件!');
  294. });
  295. } else {
  296. alert("未提取到任何作品链接!")
  297. }
  298. }, order);
  299. }
  300.  
  301. function createContainer() {
  302. let container = document.createElement('div');
  303. container.id = 'xhsFunctionContainer';
  304.  
  305. let imgTextContainer = document.createElement('div');
  306. imgTextContainer.id = 'xhsImgTextContainer';
  307.  
  308. let img = new Image(48, 48); // 确保 icon 变量已定义
  309. img.src = icon;
  310. img.style.borderRadius = '50%';
  311. img.style.objectFit = 'cover';
  312.  
  313. let textDiv = document.createElement('div');
  314. textDiv.id = 'xhsImgTextContainer__text'
  315. textDiv.textContent = 'XHS-Downloader';
  316.  
  317. imgTextContainer.appendChild(img);
  318. imgTextContainer.appendChild(textDiv);
  319.  
  320. container.appendChild(imgTextContainer);
  321.  
  322. document.body.appendChild(container);
  323. return container;
  324. }
  325.  
  326. function createButton(id, text, onClick, ...args) {
  327. let button = document.createElement('button');
  328. button.id = id;
  329. button.textContent = text;
  330. button.addEventListener('click', () => onClick(...args));
  331. return button;
  332. }
  333.  
  334. function updateContainer(buttons) {
  335. let container = document.getElementById('xhsFunctionContainer');
  336. if (!container) {
  337. container = createContainer();
  338. }
  339.  
  340. // 移除除了 imgTextContainer 以外的所有子元素
  341. Array.from(container.children).forEach(child => {
  342. if (child.id !== 'xhsImgTextContainer') {
  343. child.remove();
  344. }
  345. });
  346.  
  347. // 添加有效按钮
  348. buttons.forEach(button => {
  349. container.appendChild(button);
  350. });
  351. }
  352.  
  353. const buttons = [createButton("Download", "下载无水印作品文件", extractDownloadLinks), createButton("Post", "提取发布作品链接", extractAllLinksEvent, 0), createButton("Collection", "提取收藏作品链接", extractAllLinksEvent, 1), createButton("Favorite", "提取点赞作品链接", extractAllLinksEvent, 2), createButton("Feed", "提取发现作品链接", extractAllLinksEvent, -1),]
  354.  
  355. function run(url) {
  356. if (url === "https://www.xiaohongshu.com/explore") {
  357. updateContainer(buttons.slice(-1));
  358. } else if (url.includes("https://www.xiaohongshu.com/explore/")) {
  359. updateContainer(buttons.slice(0, 1));
  360. } else if (url.includes("https://www.xiaohongshu.com/user/profile/")) {
  361. updateContainer(buttons.slice(1, 4));
  362. }
  363. }
  364.  
  365. let currentUrl = window.location.href;
  366.  
  367. // 初始化容器
  368. run(currentUrl)
  369.  
  370. // 设置 MutationObserver 来监听 URL 变化
  371. let observer = new MutationObserver(function (mutationsList, observer) {
  372. if (currentUrl !== window.location.href) {
  373. currentUrl = window.location.href;
  374. run(currentUrl);
  375. }
  376. });
  377.  
  378. const config = {childList: true, subtree: true};
  379.  
  380. observer.observe(document.body, config);
  381.  
  382. const buttonStyle = `
  383. #xhsFunctionContainer {
  384. position: fixed;
  385. bottom: 15%;
  386. background-color: #fff;
  387. color: #2f3542;
  388. padding: 5px 10px;
  389. border-radius: 0 32px 32px 0;
  390. box-shadow: 0 3.2px 12px #00000014, 0 5px 24px #0000000a;
  391. transition: width 0.25s ease-in-out, border-radius 0.25s ease-in-out, height 0.25s ease-in-out;
  392. overflow: hidden;
  393. white-space: nowrap;
  394. width: 65px; /* 初始宽度 */
  395. height: 60px;
  396. text-align: center;
  397. font-size: 16px;
  398. display: flex;
  399. flex-direction: column-reverse;
  400. z-index: 99999;
  401. }
  402. #xhsFunctionContainer:hover {
  403. padding: 10px 10px 5px 10px;
  404. width: 210px; /* hover时的宽度 */
  405. height: auto;
  406. }
  407.  
  408. #xhsFunctionContainer button {
  409. cursor: pointer;
  410. height: 48px;
  411. color: #ff4757;
  412. font-size: 14px;
  413. font-weight: 600;
  414. border-radius: 32px;
  415. margin-bottom: 14px;
  416. border: 3px #ff4757 solid;
  417. }
  418. #xhsFunctionContainer button:active {
  419. background-color: #ff4757; /* 点击时的背景颜色 */
  420. }
  421. #xhsImgTextContainer {
  422. display: flex;
  423. align-items: center;
  424. gap: 14px;
  425. }
  426. #xhsImgTextContainer__text {
  427. font-size: 14px;
  428. font-weight: 600;
  429. }
  430. `;
  431.  
  432. const head = document.head || document.getElementsByTagName('head')[0];
  433. const style = document.createElement('style');
  434. head.appendChild(style);
  435.  
  436. style.type = 'text/css';
  437. style.appendChild(document.createTextNode(buttonStyle));
  438. })();