mirror of
https://github.com/JoeanAmier/XHS-Downloader.git
synced 2025-12-26 04:48:05 +08:00
469 lines
41 KiB
JavaScript
469 lines
41 KiB
JavaScript
// ==UserScript==
|
||
// @name XHS-Downloader
|
||
// @namespace https://github.com/JoeanAmier/XHS-Downloader
|
||
// @version 1.0
|
||
// @description 下载小红书无水印图文/视频作品文件
|
||
// @author JoeanAmier
|
||
// @match http*://www.xiaohongshu.com/explore*
|
||
// @match http*://www.xiaohongshu.com/user/profile/*
|
||
// @icon 
|
||
// @grant none
|
||
// @license GNU General Public License v3.0
|
||
// @updateURL https://github.com/JoeanAmier/XHS-Downloader/blob/master/static/XHS-Downloader.js
|
||
// @downloadURL https://github.com/JoeanAmier/XHS-Downloader/blob/master/static/XHS-Downloader.js
|
||
// ==/UserScript==
|
||
|
||
(function () {
|
||
const icon = "";
|
||
|
||
function exploreDeal(htmlSource) {
|
||
try {
|
||
let links, type_;
|
||
if (htmlSource.includes("originVideoKey")) {
|
||
type_ = "v";
|
||
links = generate_video_url(htmlSource);
|
||
} else {
|
||
type_ = "n";
|
||
links = generate_image_url(htmlSource);
|
||
}
|
||
if (links.length > 0) {
|
||
download(links, type_);
|
||
} else {
|
||
abnormal()
|
||
}
|
||
} catch (error) {
|
||
console.error("Error in deal function:", error);
|
||
abnormal();
|
||
}
|
||
}
|
||
|
||
function indexDeal() {
|
||
let headers = {
|
||
'Cookie': document.cookie, 'User-Agent': navigator.userAgent,
|
||
};
|
||
fetch(window.location.href, {
|
||
method: 'GET', headers: headers,
|
||
})
|
||
.then(response => response.text())
|
||
.then(data => {
|
||
exploreDeal(data);
|
||
})
|
||
.catch(error => {
|
||
abnormal();
|
||
});
|
||
}
|
||
|
||
function deal() {
|
||
let htmlSource = document.documentElement.outerHTML;
|
||
if (htmlSource.includes("style=\"overflow")) {
|
||
indexDeal();
|
||
} else {
|
||
exploreDeal(htmlSource);
|
||
}
|
||
}
|
||
|
||
function generate_video_url(source) {
|
||
const regex = /"originVideoKey":"(.+?)"/;
|
||
try {
|
||
let match = source.match(regex);
|
||
if (match && match[1]) {
|
||
return [decodeUnicodeString(`https://sns-video-hw.xhscdn.com/${match[1]}`)];
|
||
} else {
|
||
return []
|
||
}
|
||
} catch (error) {
|
||
console.error("Error generating video URL:", error);
|
||
return [];
|
||
}
|
||
}
|
||
|
||
function generate_image_url(source) {
|
||
const regex = /"urlDefault":"http:\\u002F\\u002Fsns-webpic-qc\.xhscdn\.com\\u002F\d+?\\u002F\S+?\\u002F(\S+?)!/g;
|
||
let matches;
|
||
let urls = [];
|
||
try {
|
||
while ((matches = regex.exec(source)) !== null) {
|
||
if (matches[1]) {
|
||
urls.push(decodeUnicodeString(`https://sns-img-bd.xhscdn.com/${matches[1]}`));
|
||
}
|
||
}
|
||
return urls
|
||
} catch (error) {
|
||
console.error("Error generating image URLs:", error);
|
||
return [];
|
||
}
|
||
}
|
||
|
||
function abnormal() {
|
||
alert("提取无水印文件下载地址失败!建议及时告知作者修复!\n项目地址:https://github.com/JoeanAmier/XHS-Downloader");
|
||
}
|
||
|
||
function decodeUnicodeString(unicodeString) {
|
||
return decodeURIComponent(JSON.parse('"' + unicodeString.replace(/"/g, '\\"') + '"'))
|
||
}
|
||
|
||
function download(urls, type_) {
|
||
if (type_ === "v") {
|
||
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 scrollToBottom(callback) {
|
||
// 滚动到页面底部的函数,调用回调函数 when finished
|
||
|
||
let lastHeight = document.body.scrollHeight;
|
||
let attempts = 0; // 添加一个计数器来避免无限循环
|
||
|
||
function scrollDown() {
|
||
// 内部函数,用于执行滚动操作
|
||
|
||
window.scrollTo(0, document.body.scrollHeight);
|
||
setTimeout(() => {
|
||
let newHeight = document.body.scrollHeight;
|
||
if (newHeight !== lastHeight) {
|
||
// 如果页面高度发生变化,则继续滚动
|
||
|
||
lastHeight = newHeight;
|
||
attempts = 0; // 重置计数器
|
||
scrollDown();
|
||
} else {
|
||
if (attempts < 3) { // 尝试几次以确认不再加载新内容
|
||
attempts++;
|
||
scrollDown();
|
||
} else {
|
||
// 停止滚动,调用回调函数
|
||
|
||
callback();
|
||
}
|
||
}
|
||
}, 2000); // 增加等待时间以确保内容加载
|
||
}
|
||
|
||
scrollDown();
|
||
}
|
||
|
||
function getFormattedURLsString() {
|
||
const notesRawValue = window.__INITIAL_STATE__.user.notes._rawValue;
|
||
|
||
let uniqueIds = new Set();
|
||
|
||
notesRawValue.forEach((notesArray) => {
|
||
notesArray.forEach((note) => {
|
||
if (note.id && note.exposed) {
|
||
uniqueIds.add(note.id);
|
||
}
|
||
});
|
||
});
|
||
|
||
let formattedURLs = [...uniqueIds].map(id => `https://www.xiaohongshu.com/explore/${id}`);
|
||
return formattedURLs.join(" ");
|
||
}
|
||
|
||
function getAllPostLinks(callback) {
|
||
scrollToBottom(() => {
|
||
const urlsString = getFormattedURLsString();
|
||
callback(urlsString);
|
||
});
|
||
}
|
||
|
||
function getAllPostLinksEvent() {
|
||
getAllPostLinks(urlsString => {
|
||
navigator.clipboard.writeText(urlsString).then(() => {
|
||
alert('链接已复制到剪贴板');
|
||
}).catch(err => {
|
||
console.error('无法复制到剪贴板', err);
|
||
});
|
||
});
|
||
}
|
||
|
||
function createContainer() {
|
||
let container = document.createElement('div');
|
||
container.id = 'xhsButtonContainer';
|
||
|
||
let imgTextContainer = document.createElement('div');
|
||
imgTextContainer.id = 'xhsImgTextContainer';
|
||
|
||
let img = new Image(28, 28); // 确保 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, condition, onClick) {
|
||
if (condition) {
|
||
let button = document.createElement('button');
|
||
button.id = id;
|
||
button.textContent = text;
|
||
button.addEventListener('click', onClick);
|
||
return button;
|
||
}
|
||
return null;
|
||
}
|
||
|
||
|
||
function updateContainer() {
|
||
let container = document.getElementById('xhsButtonContainer');
|
||
if (!container) {
|
||
container = createContainer();
|
||
}
|
||
|
||
// 移除除了 imgTextContainer 以外的所有子元素
|
||
let elementsToRemove = Array.from(container.children).filter(child => child.id !== 'xhsImgTextContainer');
|
||
elementsToRemove.forEach(element => container.removeChild(element));
|
||
|
||
|
||
// 按钮列表
|
||
let buttons = [
|
||
createButton('xhsDownloadButton', '下载无水印作品文件', window.location.href.includes("https://www.xiaohongshu.com/explore/"), deal),
|
||
createButton('xhsGetLinksButton', '获取所有帖子链接', window.location.href.includes("https://www.xiaohongshu.com/user/profile/"), getAllPostLinksEvent)
|
||
];
|
||
|
||
console.log(buttons)
|
||
|
||
// 添加有效按钮
|
||
buttons.forEach(button => {
|
||
if (button) container.appendChild(button);
|
||
});
|
||
|
||
// 当没有按钮时显示提示
|
||
if (container.children.length === 0) {
|
||
let noAction = document.createElement('div');
|
||
noAction.textContent = '无操作';
|
||
container.appendChild(noAction);
|
||
}
|
||
}
|
||
|
||
|
||
// 初始化容器
|
||
updateContainer();
|
||
|
||
let currentUrl = window.location.href;
|
||
// 设置 MutationObserver 来监听 URL 变化
|
||
let observer = new MutationObserver(mutations => {
|
||
|
||
if (window.location.href !== currentUrl) {
|
||
console.log('done')
|
||
currentUrl = window.location.href;
|
||
updateContainer();
|
||
}
|
||
});
|
||
|
||
// 配置和启动观察者
|
||
observer.observe(document.body, { childList: true, subtree: true });
|
||
|
||
})();
|
||
|
||
// ==UserScript==
|
||
// @name XHS Style
|
||
// @grant none
|
||
// ==/UserScript==
|
||
|
||
(function () {
|
||
'use strict';
|
||
|
||
const css = `
|
||
#xhsButtonContainer {
|
||
position: fixed;
|
||
bottom: 140px;
|
||
left: 0px;
|
||
background-color: #fff;
|
||
color: #000;
|
||
padding: 5px 10px;
|
||
border-radius: 0 28px 28px 0;
|
||
box-shadow: 0 3.2px 12px #00000014, 0 5px 25px #0000000a;
|
||
transition: width 0.3s ease-in-out, border-radius 0.3s ease-in-out, height 0.3s ease-in-out;
|
||
overflow: hidden;
|
||
white-space: nowrap;
|
||
width: 50px; /* 初始宽度 */
|
||
height: 40px;
|
||
text-align: center;
|
||
font-size: 16px;
|
||
display: flex;
|
||
flex-direction: column-reverse;
|
||
z-index: 1000;
|
||
}
|
||
|
||
#xhsButtonContainer:hover {
|
||
padding: 10px 10px 5px 10px;
|
||
width: 170px; /* hover时的宽度 */
|
||
height: auto;
|
||
}
|
||
|
||
#xhsButtonContainer button {
|
||
cursor: pointer;
|
||
height: 32px;
|
||
width: 145px;
|
||
color: #E53936;
|
||
font-size: 14px;
|
||
font-weight: 600;
|
||
border-radius: 32px;
|
||
margin-bottom: 14px;
|
||
border: 2px #E53936 solid;
|
||
}
|
||
|
||
#xhsButtonContainer button:active {
|
||
background-color: #E53936; /* 点击时的背景颜色 */
|
||
}
|
||
|
||
#xhsImgTextContainer {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 12px;
|
||
}
|
||
|
||
#xhsImgTextContainer__text {
|
||
font-size: 12px;
|
||
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(css));
|
||
})();
|