qwq
// ==UserScript==
// @name SPD测试脚本Mod
// @namespace speedownload
// @version 2025-10-14
// @description qwq
// @match *://pan.baidu.com/disk/main*
// @icon https://nd-static.bdstatic.com/m-static/v20-main/favicon-main.ico
// @grant GM_xmlhttpRequest
// @grant unsafeWindow
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_download
// @grant GM_info
// @grant GM_registerMenuCommand
// @connect localhost
// @connect 127.0.0.1
// @connect pcs.baidu.com
// @connect baidupcs.com
// @connect *.baidupcs.com
// @connect speedownload.com
// @connect *.speedownload.com
// @connect *
// ==/UserScript==
(function() {
'use strict';
// 默认配置
const defaultConfig = {
method: 0, // 0 = 浏览器下载 | 1 = Aria2下载 | 2 = AB Download Manager
aria2: {
host: 'http://127.0.0.1:6800/jsonrpc',
token: '',
defaultDownloadPath: 'F:/',
},
abdmHost: 'http://localhost:15151',
abdmSilent: false,
accelerateUrl: "你的加速链接",
};
// 从存储中获取配置
function getConfig() {
const config = { ...defaultConfig };
config.method = GM_getValue('method', defaultConfig.method);
config.aria2.host = GM_getValue('aria2_host', defaultConfig.aria2.host);
config.aria2.token = GM_getValue('aria2_token', defaultConfig.aria2.token);
config.aria2.defaultDownloadPath = GM_getValue('aria2_path', defaultConfig.aria2.defaultDownloadPath);
config.abdmHost = GM_getValue('abdm_host', defaultConfig.abdmHost),
config.abdmSilent = GM_getValue('abdm_silent', defaultConfig.abdmSilent),
config.accelerateUrl = GM_getValue('accelerate_url', defaultConfig.accelerateUrl);
return config;
}
// 保存配置到存储
function saveConfig(config) {
GM_setValue('method', config.method);
GM_setValue('aria2_host', config.aria2.host);
GM_setValue('aria2_token', config.aria2.token);
GM_setValue('aria2_path', config.aria2.defaultDownloadPath);
GM_setValue('abdm_host', config.abdmHost);
GM_setValue('abdm_silent', config.abdmSilent);
GM_setValue('accelerate_url', config.accelerateUrl);
}
// 创建配置界面
function createConfigDialog() {
let Vue = this;
const config = getConfig();
// 移除已存在的配置窗口
const existingDialog = document.getElementById('spd-config-dialog');
if (existingDialog) {
existingDialog.remove();
}
// 创建遮罩层
const overlay = document.createElement('div');
overlay.id = 'spd-config-dialog';
overlay.style.cssText = `
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(0, 0, 0, 0.5);
z-index: 10000;
display: flex;
justify-content: center;
align-items: center;
`;
// 创建配置窗口
const dialog = document.createElement('div');
dialog.style.cssText = `
background: white;
border-radius: 8px;
padding: 20px;
width: 500px;
max-width: 90vw;
max-height: 80vh;
overflow-y: auto;
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.3);
`;
dialog.innerHTML = `
<div style="margin-bottom: 20px;">
<h2 style="margin: 0 0 15px 0; color: #333; font-size: 18px;">SPD下载脚本配置</h2>
<div style="margin-bottom: 15px;">
<label style="display: block; margin-bottom: 5px; font-weight: bold;">下载方式:</label>
<select id="spd-method" style="width: 100%; padding: 8px; border: 1px solid #ddd; border-radius: 4px;">
<option value="0" ${config.method === 0 ? 'selected' : ''}>浏览器下载</option>
<option value="1" ${config.method === 1 ? 'selected' : ''}>Aria2下载</option>
<option value="2" ${config.method === 2 ? 'selected' : ''}>AB Download Manager</option>
</select>
</div>
<div style="margin-bottom: 15px;">
<label style="display: block; margin-bottom: 5px; font-weight: bold;">加速链接:</label>
<input type="text" id="spd-accelerate-url" value="${config.accelerateUrl}"
style="width: 100%; padding: 8px; border: 1px solid #ddd; border-radius: 4px; box-sizing: border-box;"
placeholder="请输入加速链接">
</div>
<fieldset id="aria2-config" style="border: 1px solid #ddd; border-radius: 4px; padding: 15px; margin-bottom: 15px; ${config.method === 1 ? 'display: block;' : 'display: none;'}">
<legend style="font-weight: bold; color: #666;">Aria2配置</legend>
<div style="margin-bottom: 10px;">
<label style="display: block; margin-bottom: 5px;">Aria2 RPC地址:</label>
<input type="text" id="spd-aria2-host" value="${config.aria2.host}"
style="width: 100%; padding: 8px; border: 1px solid #ddd; border-radius: 4px; box-sizing: border-box;"
placeholder="http://127.0.0.1:6800/jsonrpc">
</div>
<div style="margin-bottom: 10px;">
<label style="display: block; margin-bottom: 5px;">Aria2 Token:</label>
<input type="text" id="spd-aria2-token" value="${config.aria2.token}"
style="width: 100%; padding: 8px; border: 1px solid #ddd; border-radius: 4px; box-sizing: border-box;"
placeholder="留空如果没有设置token">
</div>
<div style="margin-bottom: 10px;">
<label style="display: block; margin-bottom: 5px;">默认下载路径:</label>
<input type="text" id="spd-aria2-path" value="${config.aria2.defaultDownloadPath}"
style="width: 100%; padding: 8px; border: 1px solid #ddd; border-radius: 4px; box-sizing: border-box;"
placeholder="F:/">
<small style="color: #666; font-size: 12px;">注意:路径格式如 D:/ 或 /home/user/downloads/</small>
</div>
</fieldset>
<fieldset id="abdm-config" style="border: 1px solid #ddd; border-radius: 4px; padding: 15px; margin-bottom: 15px; ${config.method === 2 ? 'display: block;' : 'display: none;'};">
<div style="margin-bottom: 15px;">
<label style="display: block; margin-bottom: 5px; font-weight: bold;">AB Download Manager 接口地址:</label>
<input type="text" id="spd-abdm-host" value="${config.abdmHost}"
style="width: 100%; padding: 8px; border: 1px solid #ddd; border-radius: 4px; box-sizing: border-box;"
placeholder="http://localhost:15151/start-headless-download">
</div>
<div style="margin-bottom: 10px;">
<label>
<input type="checkbox" id="spd-abdm-silent" ${config.abdmSilent ? 'checked' : ''}> 启用静默下载(跳过确认界面)
</label>
</div>
</fieldset>
<div style="display: flex; gap: 10px; justify-content: flex-end;">
<button id="spd-config-cancel" style="padding: 8px 16px; border: 1px solid #ddd; background: #f5f5f5; border-radius: 4px; cursor: pointer;">取消</button>
<button id="spd-config-save" style="padding: 8px 16px; border: none; background: #007cff; color: white; border-radius: 4px; cursor: pointer;">保存</button>
</div>
</div>
`;
overlay.appendChild(dialog);
document.body.appendChild(overlay);
// 获取元素
const methodSelect = document.getElementById('spd-method');
const aria2Config = document.getElementById('aria2-config');
const abdmConfig = document.getElementById('abdm-config');
// 下载方式切换事件
methodSelect.addEventListener('change', function() {
const selectedMethod = parseInt(this.value);
if(selectedMethod === 2){
aria2Config.style.display = 'none';
aria2Config.style.opacity = '0';
abdmConfig.style.display = 'block';
abdmConfig.style.opacity = '1';
}else if (selectedMethod === 1) {
abdmConfig.style.display = 'none';
abdmConfig.style.opacity = '0';
aria2Config.style.display = 'block';
aria2Config.style.opacity = '1';
} else {
// 选择浏览器下载,隐藏Aria2配置
aria2Config.style.opacity = '0';
aria2Config.style.display = 'none';
abdmConfig.style.opacity = '0';
abdmConfig.style.display = 'none';
}
});
// 绑定事件
document.getElementById('spd-config-cancel').onclick = () => {
overlay.remove();
};
document.getElementById('spd-config-save').onclick = () => {
const newConfig = {
method: parseInt(document.getElementById('spd-method').value),
aria2: {
host: document.getElementById('spd-aria2-host').value.trim(),
token: document.getElementById('spd-aria2-token').value.trim(),
defaultDownloadPath: document.getElementById('spd-aria2-path').value.trim(),
},
abdmHost: document.getElementById('spd-abdm-host').value.trim(),
abdmSilent: document.getElementById('spd-abdm-silent')?.checked ?? false,
accelerateUrl: document.getElementById('spd-accelerate-url').value.trim()
};
// 验证配置
if (!newConfig.accelerateUrl) {
alert('请输入加速链接');
return;
}
if (newConfig.method === 1) {
if (!newConfig.aria2.host) {
alert('请输入Aria2 RPC地址');
return;
}
if (!newConfig.aria2.defaultDownloadPath) {
alert('请输入默认下载路径');
return;
}
// 验证RPC地址格式
if (!newConfig.aria2.host.startsWith('http://') && !newConfig.aria2.host.startsWith('https://')) {
alert('Aria2 RPC地址必须以 http:// 或 https:// 开头');
return;
}
}
if (newConfig.method === 2) {
if (!newConfig.abdmHost) {
alert('请输入AB Download Manager API地址');
return;
}
// 验证RPC地址格式
if (!newConfig.abdmHost.startsWith('http://') && !newConfig.abdmHost.startsWith('https://')) {
alert('Aria2 RPC地址必须以 http:// 或 https:// 开头');
return;
}
}
saveConfig(newConfig);
// 显示保存成功的样式提示
const saveBtn = document.getElementById('spd-config-save');
const originalText = saveBtn.innerHTML;
saveBtn.innerHTML = '✓ 保存成功';
saveBtn.style.background = '#28a745';
setTimeout(() => {
saveBtn.innerHTML = originalText;
saveBtn.style.background = '#007cff';
overlay.remove();
}, 300);
};
// 点击遮罩层关闭
// overlay.onclick = (e) => {
// if (e.target === overlay) {
// overlay.remove();
// }
// };
}
// 创建设置按钮
function createSettingsButton() {
// 避免重复创建
if (document.getElementById('spd-settings-btn')) {
return;
}
const settingsBtn = document.createElement('button');
settingsBtn.id = 'spd-settings-btn';
settingsBtn.innerHTML = '⚙️ SPD设置';
settingsBtn.style.cssText = `
position: fixed;
top: 10px;
right: 10px;
z-index: 9999;
padding: 8px 12px;
background: #007cff;
color: white;
border: none;
border-radius: 4px;
cursor: pointer;
font-size: 12px;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.2);
transition: background 0.3s ease;
`;
// 悬停效果
settingsBtn.onmouseenter = () => {
settingsBtn.style.background = '#0056cc';
};
settingsBtn.onmouseleave = () => {
settingsBtn.style.background = '#007cff';
};
settingsBtn.onclick = createConfigDialog;
document.body.appendChild(settingsBtn);
}
// 创建侧边栏设置按钮
function createSidebarSettingsButton() {
// 避免重复创建
if (document.querySelector('.spd-settings-sidebar-btn')) {
return;
}
// 等待侧边栏加载完成
const checkSidebar = setInterval(() => {
const sidebar = document.querySelector('.wp-s-aside-nav__main-top');
if (sidebar) {
clearInterval(checkSidebar);
// 创建SPD设置按钮
const settingsBtn = document.createElement('a');
settingsBtn.href = 'javascript:;';
settingsBtn.className = 'u-tooltip item wp-s-aside-nav__main-item wp-s-aside-nav__main-item spd-settings-sidebar-btn';
settingsBtn.innerHTML = `
<div class="wp-s-aside-nav__main-item-wrap">
<img src="" alt="" class="wp-s-aside-nav__main-item-img">
<img src="" alt="" class="wp-s-aside-nav__main-item-img is-active-img">
<p class="wp-s-aside-nav__main-item-text text-ellip">
SPD设置
</p>
</div>
`;
settingsBtn.style.cssText = `
cursor: pointer;
`;
settingsBtn.onclick = createConfigDialog;
// 插入到"消息"按钮后面
sidebar.appendChild(settingsBtn);
}
}, 200);
}
// 立即注册(不可用)菜单命令
GM_registerMenuCommand('⚙️ 下载设置', createConfigDialog);
// 获取当前配置
function getCurrentConfig() {
return getConfig();
}
function parseResponseHeaders(headersString) {
const headers = {};
if (!headersString) return headers;
const headerPairs = headersString.split('\n');
for (const header of headerPairs) {
const index = header.indexOf(':');
if (index > 0) {
const key = header.substring(0, index).trim();
const value = header.substring(index + 1).trim();
headers[key] = value;
}
}
return headers;
}
function downloadFile(method, url, info, ua) {
let Vue = this;
if(method == 0){
let lastUpdateTime = 0;
GM_download({
url: url,
name: decodeURIComponent(info.server_filename || info.fileName),
headers: {
'User-Agent': ua
},
onprogress: function (progress) {
if (progress.total) {
const now = Date.now();
if (now - lastUpdateTime > 1000) { // 每秒更新一次
lastUpdateTime = now;
let percent = ((progress.loaded / progress.total) * 100).toFixed(2);
Vue.$svipMessage.closeAll(); // 关闭所有消息
Vue.$svipMessage({
message: `正在下载 ${info.server_filename}:${percent}%`,
type: "info",
duration: 0, // 不自动关闭
showClose: false // 显示关闭按钮
});
}
}
},
onload: function(response) {
// 关闭进度消息
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: `文件 ${info.server_filename} 下载完成`,
type: 'success',
duration: 2000,
});
},
onerror: function(error) {
// 关闭进度消息
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: `文件 ${info.server_filename} 下载失败: ${error.error}`,
type: 'error',
duration: 3000,
});
}
});
} else if(method == 1){
GM_xmlhttpRequest({
method: "HEAD",
url: url,
onload: function (response) {
const config = getCurrentConfig();
let dirPath = info.path.split('/').slice(0, -1).join('/');
let downloadDir = config.aria2.defaultDownloadPath.replaceAll("\\", "/") + dirPath;
let data = {
id: new Date().getTime(),
jsonrpc: '2.0',
method: 'aria2.addUri',
params: [`token:${config.aria2.token}`, [response.finalUrl], {
'dir': downloadDir,
'out': decodeURIComponent(info.server_filename || info.fileName),
'max-connection-per-server': 16,
'header': [
`User-Agent: ${ua}`,
]
}]
};
GM_xmlhttpRequest({
method: "POST",
url: config.aria2.host,
data: JSON.stringify(data),
onload: function () {
console.log("Aria2 下载任务已提交:", info.server_filename);
},
onerror: function (error) {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: "无法把文件推送至Aria2,请检查Aria2状态",
type: "error",
duration: 3000
});
}
});
},
onerror: function (error) {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: "无法获取 " + info.server_filename + " 的下载链接",
type: "error",
duration: 3000
});
}
});
} else if (method == 2) {
// AB Download Manager下载方式
const config = getCurrentConfig()
let abdmHost = config.abdmHost.replace(/\/+$/, '');
let endpoint = config.abdmSilent ? '/start-headless-download' : '/add';
// 构造请求体
let data = null;
if (config.abdmSilent) {
data = {
downloadSource: [{
link: url,
headers: ua ? { "User-Agent": ua } : {}
// downloadPage: window.location.href
}],
// folder: '', // 可以允许用户配置,也可以留空
name: info.server_filename || '',
queueId: 0 // 可选,默认0即无队列
};
} else {
data = [{
link: url,
headers: ua ? { "User-Agent": ua } : {}
// downloadPage: window.location.href
}];
}
GM_xmlhttpRequest({
method: 'POST',
url: abdmHost + endpoint,
headers: {
'Content-Type': 'application/json'
},
data: JSON.stringify(data),
onload: function (response) {
console.log("发送abdm", response)
if (response.status === 200) {
Vue.$svipMessage({
message: `任务已发送到 AB Download Manager: ${data.name}`,
type: 'success',
duration: 3000,
});
} else {
Vue.$svipMessage({
message: `发送到 AB Download Manager 失败,状态码:${response.status}`,
type: 'error',
duration: 3000,
});
}
},
onerror: function (error) {
console.log(error);
Vue.$svipMessage({
message: '无法连接 AB Download Manager,请确保其已启动并监听接口',
type: 'error',
duration: 3000,
});
}
});
} else {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
type: "error",
message: "无效的下载方式,请进入脚本编辑页重新设置",
duration: 3000,
});
return;
}
}
function accelerateDownload(file) {
let Vue = this;
const config = getCurrentConfig();
Vue.dlinkIns.getSign().then(function () {
GM_xmlhttpRequest({
method: "POST",
url: "/api/download?clienttype=8&app_id=250528&web=1&fidlist=["+file.fs_id+"]&type=dlink&sign=" + encodeURIComponent(Vue.dlinkIns.sign) + "×tamp=" + Vue.dlinkIns.timestamp,
onload: function (response) {
console.log("请求下载/api/download 响应",JSON.parse(response.response))
let dlink = JSON.parse(response.response).dlink[0].dlink
GM_xmlhttpRequest({
method: "HEAD",
url: dlink,
headers: {
"User-Agent": "pan.baidu.com"
},
onload: function (response) {
console.log("请求dlink链接的响应",response)
const headers = parseResponseHeaders(response.responseHeaders);
console.log(JSON.stringify({
md5: headers['Content-MD5'],
size: headers['Content-Length'],
name: file.server_filename,
id: file.fs_id.toString()
}))
GM_xmlhttpRequest({
method: "POST",
url: config.accelerateUrl,
data: JSON.stringify({
md5: headers['Content-MD5'],
size: headers['Content-Length'],
name: file.server_filename,
id: file.fs_id.toString()
}),
headers: {
"Content-Type": "application/json"
},
onload: function (response) {
console.log("请求加速链接响应",response)
let res = JSON.parse(response.response)
if(res.code != 0) {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: res.message,
type: "error",
duration: 2000,
});
return;
}
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: res.message,
type: "sucess",
duration: 2000,
});
let url = res.data.urls[0].url
let ua = res.data.ua
downloadFile.call(Vue, config.method, url, file, ua);
},
onerror: function (error) {
console.log("请求加速链接响应error",error)
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: "请求加速链接失败",
type: "error",
duration: 3000,
});
}
});
},
onerror: function (error) {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: "无权限获取此文件信息",
type: "error",
duration: 3000
});
}
});
},
onerror: function () {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: "无法请求百度网盘下载服务器",
type: "error",
duration: 3000,
});
}
});
});
}
function download(dir) {
let Vue = this;
if(dir != undefined) {
GM_xmlhttpRequest({
method: "GET",
url: "/api/list?clienttype=0&app_id=250528&web=1&order=name&desc=1&dir=" + encodeURIComponent(dir.path) + "&num=100000&page=1",
onload: function (response) {
let list = JSON.parse(response.response).list;
if(list.length < 1) {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: "这是一个空的文件夹!",
type: "error"
});
return;
}
list.forEach((file) => {
if(file.isdir == 1) {
download.call(Vue, file);
} else {
accelerateDownload.call(Vue,file);
}
});
},
onerror: function () {
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: "无法获取 " + dir.server_filename + " 内的文件",
type: "error",
duration: 3000,
});
}
});
return;
}
this.canShowFail = true;
if(this.currentFileMetas.length < 1) {
Vue.$svipMessage({
message: "请至少选择一个文件进行下载",
type: "error"
});
return;
}
if (this.currentFileMetas.length >= 1) {
const fileName = this.currentFileMetas[0].server_filename;
const msg = this.currentFileMetas.length === 1 ? `正在下载 ${fileName}` : `正在下载 ${fileName} 等其他文件...`;
Vue.$svipMessage.closeAll();
Vue.$svipMessage({
message: msg,
type: "success",
duration: 2000,
});
}
this.currentFileMetas.forEach(file => {
if(file.isdir == 1) {
download.call(Vue, file);
} else {
console.log(this.dlinkIns,file.md5,file.size,file.server_filename)
accelerateDownload.call(Vue,file);
}
})
}
let HookBaiduNetdisk = new MutationObserver(mutationRecords => {
for (let mr of mutationRecords) {
for (let node of mr.addedNodes) {
if (node.__vue__ !== undefined && node.__vue__ !== null) {
let Vue = node.__vue__;
if (Vue.$el.className == "nd-download") {
Vue.download = download;
Vue.$options.methods.download = download;
} // 重写下载方法
}
}
}
});
HookBaiduNetdisk.observe(document, {
childList: true,
subtree: true
});
// 等待页面加载后创建设置按钮
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', createSidebarSettingsButton);
} else {
setTimeout(createSidebarSettingsButton, 1000);
}
})();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址