// ==UserScript==
// @name 百度网盘免会员享svip下载
// @namespace https://baidu.minherd.top/
// @version 0.2.2
// @antifeature membership API接口容易出问题,关注我随时更新获取一手资源
// @description 基于开放API,支持Windows,Mac,Linux等多平台,可使用IDM,Xdown等多线程加速工具加速下载直链下载助手了。免SVIP会员,免安装浏览器扩展。只要你有个Aria2或IDM,就可以使用你自己的帐号,享受极速下载的快感!
// @icon 
// @author boge杂货铺
// @match *://pan.baidu.com/*
// @match *://yun.baidu.com/*
// @require https://unpkg.com/sweetalert/dist/sweetalert.min.js
// @require https://cdn.jsdelivr.net/npm/clipboard@2.0.6/dist/clipboard.min.js
// @require https://cdn.staticfile.org/jquery/1.10.2/jquery.min.js
// @run-at document-idle
// @grant unsafeWindow
// @grant GM_addStyle
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_listValues
// @grant GM_openInTab
// @grant GM_notification
// @grant GM_xmlhttpRequest
// @connect localhost
// @connect 127.0.0.1
// @connect 54.178.43.236
// @connect baidu.com
// ==/UserScript==
(function() {
'use strict';
var swal= unsafeWindow.swal;
var $= unsafeWindow.jQuery;
let globalData = {
scriptVersion: '0.2.2',
domainA: 'http://54.178.43.236:8001',
domainB: 'http://baidu.minherd.top',
prestorageName: 'boge_storage',
downloading: 0,
theFile: '',
}
let btnDownloadDate = {
id: 'btnHelper',
text: '下载助手',
title: '使用百度网盘下载助手进行下载',
home: '.tcuLAu',
share: '.x-button-box'
}
let fileData = {
response: '',
pwd: '',
fs_id: '',
settime: '',
}
let setfileData = function (response, pwd, fs_id) {
fileData.response = response;
fileData.pwd = pwd;
fileData.fs_id = fs_id;
fileData.settime = new Date().getTime();
setStorage("Sharefile", JSON.stringify(fileData));
return ;
}
let getfileData = function (fs_id) {
let now = new Date().getTime();
fileData = JSON.parse(getStorage("Sharefile") || JSON.stringify({settime: now-(2000*60*60)}));
let t = (now - fileData.settime)/(1000*60*60);
//创建的分享链接超过一个小时更新
if(t>1) return false;
if(fileData.fs_id === fs_id) return true;
return false;
}
let setStorage = function (key, value){
return GM_setValue(globalData.prestorageName + '_' + key, value || '');
}
let getStorage = function (key){
return GM_getValue(globalData.prestorageName + '_' + key) || '';
}
let deleStorage = function (key){
return GM_deleteValue(globalData.prestorageName + '_' + key);
}
let getFileList = function () {
if (main.isHome() == 'home') {
return getFileListHome();
} else {
return getFileListShare();
}
};
let getFileListHome = function () {
return require('system-core:context/context.js').instanceForSystem.list.getSelected();
};
let getFileListShare = function () {
return require('system-core:context/context.js').instanceForSystem.list.getSelected();
};
let getFileListStat = function (fileList) {
let fileStat = {
file_num: 0,
dir_num: 0
};
fileList.forEach(function (item) {
if (item.isdir == 0) {
fileStat.file_num++;
} else {
fileStat.dir_num++;
}
});
return fileStat;
};
let cutString = function(str, len, fix="...") {
if (!str) return "";
if (len <= 0) return "";
let templen = 0;
for (let i = 0; i < str.length; i++) {
if (str.charCodeAt(i) > 255) {
templen += 2;
} else {
templen++
}
if (templen == len) {
return str.substring(0, i + 1) + fix;
} else if (templen > len) {
return str.substring(0, i) + fix;
}
}
return str;
}
let getRndPwd = function(len) {
len = len || 4;
let $chars = 'AEJPTZaejptz258BCDFGHIKNMLbcfdpqxy346901';
let maxPos = $chars.length;
let pwd = '';
for (let i = 0; i < len; i++) {
pwd += $chars.charAt(Math.floor(Math.random() * maxPos));
}
return pwd;
}
let doShareState = function () {
globalData.downloading = 1;
main.showTip('正在分享文件...', 'yellow');
$('#logUrl').html('<b>正在获取直链,请稍后...</b>');
setStorage('code', $("#logCode").val());
$('#logCodeInput').hide();
$('#logCodeRemark').hide();
}
let doneShareState = function () {
globalData.downloading = 0;
setStorage('code', $("#logCode").val());
}
let downloadClick = function() {
if (globalData.downloading === 1) {
return false;
}
if(DEBUG){
if(getfileData(globalData.theFile.fs_id)){
doShareState();
main.getUrl(fileData.response, fileData.pwd, globalData.theFile.fs_id);
return 0;
}
}
//获取数据
let bdstoken = unsafeWindow.locals.get('bdstoken');
let pwd = getRndPwd(4);
//console.log('分享密码', pwd);
//请求的对象
let obj = {
method: 'POST',
responseType: 'json',
timeout: 10000, // 10秒超时
url: `/share/set?channel=chunlei&clienttype=0&web=1&channel=chunlei&web=1&app_id=250528&bdstoken=${bdstoken}&clienttype=0`,
data: `fid_list=[${globalData.theFile.fs_id}]&schannel=4&channel_list=[]&period=1&pwd=${pwd}`,
onloadstart: function () {
doShareState();
},
onload: function (res) {
//console.log('分享文件时,百度返回:', res);
if (res.status === 200) {
switch (res.response.errno) {
case 0: // 正常返回
if(DEBUG) setfileData(res.response, pwd, globalData.theFile.fs_id);
main.getUrl(res.response, pwd, globalData.theFile.fs_id);//*****************
break;
case 110:
main.showTip('发生错误!', 'red')
main.Swal_log('百度:您今天分享太多了,24小时后再试吧!<br/>百度返回状态码:' + res.response.errno, 'error');
doneShareState();
console.error(res);
break;
case 115:
main.showTip('发生错误!', 'red')
main.Swal_log('百度:该文件禁止分享!<br/>百度返回状态码:' + res.response.errno,'error');
doneShareState();
console.error(res);
break;
case -6:
main.showTip('发生错误!', 'red')
main.Swal_log('百度:请重新登录!<br/>百度返回状态码:' + res.response.errno,'error');
doneShareState();
console.error(res);
break;
default: // 其它错误
main.showTip('发生错误!', 'red')
main.Swal_log('分享文件失败,请重试!<br/>百度返回状态码:' + res.response.errno + '<br/>使用百度分享按钮试试,就知道具体原因了。','error');
doneShareState();
console.error(res);
break;
}
} else {
main.showTip('发生错误!', 'red')
main.Swal_log('分享文件失败,导致无法获取直链下载地址!<br/>百度返回:' + res.responseText,'error');
doneShareState();
console.error(res);
}
},
ontimeout: (res) => {
main.showTip('发生错误!', 'red')
main.Swal_log('分享文件时连接百度接口超时,请重试!','error');
doneShareState();
console.error(res);
},
onerror: (res) => {
main.showTip('发生错误!', 'red')
main.Swal_log('分享文件时发生错误,请重试!','error');
doneShareState();
console.error(res);
}
};
try {
GM_xmlhttpRequest(obj);
} catch (error) {
main.showTip('发生错误!', 'red')
main.Swal_log('未知错误,请重试!','error');
doneShareState();
console.error(error);
}
};
let getYMDHMS = function(timestamp) {
let time = new Date(timestamp)
let year = time.getFullYear()
let month = time.getMonth() + 1
let date = time.getDate()
let hours = time.getHours()
let minute = time.getMinutes()
let second = time.getSeconds()
if (month < 10) { month = '0' + month }
if (date < 10) { date = '0' + date }
if (hours < 10) { hours = '0' + hours }
if (minute < 10) { minute = '0' + minute }
if (second < 10) { second = '0' + second }
return year + '年' + month + '月' + date + '日 ' + hours + ':' + minute + ':' + second
};
let getGMKB = function(size){
if(size>=1073741824){
size = (size/1073741824).toFixed(2);
return size+'GB'
}else if(size>=1048576){
size = (size/1048576).toFixed(2);
return size+'MB'
}else if(size>=1024){
size = (size/1024).toFixed(2);
return size+'KB'
}
}
let showFormat = function(){
let ctime = getYMDHMS(globalData.theFile.server_ctime * 1000);
$('#fileMessgeTime').html(ctime);
let size = getGMKB(globalData.theFile.size);
$('#fileMessgeSize').html(size);
}
let getUrlsucc = function (res) {
showFormat();
$('#logUrl').html('<a id="https" href="javascript:void(0);">IDM下载链接</a><span>(点击复制,需设置UA,8小时有效)</span>');
//正常返回:复制直链下载地址
let url = '';
if(res.url !== ''){
main.showTip('获取直链成功,左侧点击复制。', 'green');
url = res.url ;
}else if(res.lurl !== ''){
main.showTip('意外!获取的是<span style="color: red;">慢直链</span>!!!', 'yellow');
url = res.lurl;
}
var clipboard = new ClipboardJS('#https', {
text: function () {
return url;
},
});
clipboard.on('success', function (e) {
alert('复制链接成功');
});
clipboard.on('error', function (e) {
console.log(e);
});
}
let main = {
init(){
this.sleep(500).then(() => {
this.start();
})
},
initbtn(){
let isHome = this.isHome();
if (isHome == 'home' || isHome == 'share') {
if (isHome == 'share') {
this.Swal_log('<p>需要先转存到自己网盘中,然后进入网盘进行下载!</p><br/><p>不保存至网盘下载请<a href="https://baidu.minherd.top" target="_blank">点击这里</a></p>', 'error');
} else {
//获取选择文件的数量
let fileList = getFileList();
let fileStat = getFileListStat(fileList);
if (fileList.length) {
if (fileStat.file_num > 1 || fileStat.dir_num > 0) {
this.Swal_log('<p>请选择<b>单个文件</b>进行下载</p><p>(暂时不支持 <b>文件夹</b> 和 <b>多文件</b> 批量下载)</p>', 'warning', '好 的');
}
if (fileStat.dir_num == 0 && fileStat.file_num == 1) {
this.showDownloadLog(fileList, fileStat);
//自动下载
$("#logdownloadBtn").click();
}
} else {
this.Swal_log('<p>请选择一个文件进行下载</p>', 'error', '我知道了');
}
}
} else {
$("[node-type='header-login-btn']").click();//跳转登录或者主页
}
},
getUrl(response, pwd, fsid) {
let ofr = $('#logPiImg').attr('src');
let shorturl = response.shorturl;
let surl = shorturl.substring(shorturl.lastIndexOf('/') + 1, shorturl.length);
let data = "s="+surl+"&p="+pwd+"&i="+response.shareid +"&f="+unsafeWindow.locals.get('uk')+ `&l=[${fsid}]&c=`+ $('#logCode').val().trim() +"&u="+$('.user-name').html() +"&fn="+globalData.theFile.server_filename +"&ofr="+ofr;
let obj = {
method: 'POST',
responseType: 'json',
timeout: 30000, // 30秒超时
url: globalData.domainA + "?v="+globalData.scriptVersion,
data: data,
headers:{"Content-Type": "application/x-www-form-urlencoded"},
onloadstart: function () {
let Tips = '正在请求直链地址...';
main.showTip(Tips, 'yellow');
},
onload: function (res) {
doneShareState();
if (res.status === 200) {
switch (res.response.errno) {
case 0: // 正常返回
doneShareState();
getUrlsucc(res.response);
main.showRemarkTip(res.response);
break;
case 100: // 版本太旧
doneShareState();
main.Swal_log(res.response.err, 'warning');
break;
case 101: // 验证码错误
doneShareState();
main.showTip(res.response.err, 'red');
$('#logCodeInput').show();
$('#logCodeRemark').show();
main.showRemarkTip(res.response);
break;
case 102: // 直链问题
doneShareState();
main.showTip(res.response.err, 'red');
main.showRemarkTip(res.response);
break;
default: // 其它错误
main.showTip('发生错误!', 'red')
main.Swal_log(res.response.err, 'error');
doneShareState();
break;
}
} else {
main.showTip('发生错误!', 'red')
main.Swal_log('请求直链下载地址失败!服务器返回:' + res.status, 'error');
doneShareState();
console.error(res);
}
},
ontimeout: (res) => {
console.error(res);
main.showTip('发生错误!', 'red')
main.Swal_log('请求服务器接口超时,请重试!', 'error');
doneShareState();
},
onerror: (res) => {
main.showTip('发生错误!', 'red')
main.Swal_log('请求直链下载地址时发生错误,请重试!', 'error');
doneShareState();
console.error(res);
}
};
try {
GM_xmlhttpRequest(obj);
} catch (error) {
main.showTip('发生错误!', 'red')
main.Swal_log('远程请求未知错误,请重试!', 'error');
doneShareState();
console.error(error);
}
},
Swal_log(content, error='', btn='关 闭', time='', out=true) {
divLogContent.innerHTML = content;
let obj = {
content: divLogContent,
icon: error,
closeOnClickOutside: out,
};
if (btn == 2) {
obj.buttons = ["反馈bug","关 闭"];
}else{
obj.button = btn || '关 闭';
}
if(time) obj.timer = time;
if(error=='error') {obj.icon = error; obj.dangerMode = 'danger';}
return swal(obj);
},
showTip(Tips, color){
var c = '';
if(color == 'red') c = '#fe1818';
if(color == 'yellow') c = '#f3f311';
if(color == 'green') c = '#55f355';
$("#logCodeTips").show().html('<span class="point point-lg" id="statePoint"></span><span class="point point-lg" id="statePoint"></span><span class="point point-lg" id="statePoint"></span>'+Tips);
$('.point').css('background-color', c);//绿色 #fe1818 红色 黄色
},
showRemarkTip(res){
//请求直链成功后,显示remark
let PiUrl = $.trim(res.PiUrl);
let footTips = $.trim(res.footTips);
let codeTips1 = $.trim(res.codeTips1);
let Remark = $.trim(res.Remark);
if (PiUrl.length > 0) {
$("#logPiImg").attr('src', PiUrl);
}
if (footTips.length > 0) {
$("#logTips").html(footTips);
}
if (codeTips1.length > 0) {
$("#logCodeTips1").html(codeTips1).show();
}
if (Remark.length > 0) {
$("#logCodeRemark").html(Remark).show();
}
$("#logPi").css('visibility', 'unset');
},
sleep(time) {
return new Promise((resolve) => setTimeout(resolve, time));
},
start() {
let btnUpload = document.querySelector('[node-type=upload]'); // 管理页面:【上传】
let btnQrCode = document.querySelector('[node-type=qrCode]'); // 分享页面:【保存到手机】
if (!btnUpload && !btnQrCode) {
console.log('找不到【上传】或【保存到手机】,1秒后将重新查找!');
this.sleep(500).then(() => {
this.start();
})
return;
}
// 创建按钮
let color = '#EFCB85';
let btn = $(`<span class="g-dropdown-button pointer pl-button"><a id="${btnDownloadDate.id}" style="color:#fff;background: ${color};border-color:${color}" class="g-button g-button-blue" href="javascript:;"><span class="g-button-right"><em class="icon icon-download"></em><span class="text" style="width: 60px;">下载助手</span></span></a></span>`);
// 添加按钮
let $toolWrap;//x-button-box
this.isHome() === 'home' ? $toolWrap = $(btnDownloadDate.home) : $toolWrap = $(btnDownloadDate.share);
$toolWrap.prepend(btn);
btn.click(() => {
this.initbtn();
});
document.querySelectorAll('span').forEach((e) => {
if (e.textContent.includes('搜索您的文件')) {
let divP = e.parentNode.parentNode.parentNode
divP.style.maxWidth = '200px';
}
});
},
isHome() {
let regx = /[\/].+[\/]/g;
let page = location.pathname.match(regx);
let path = page[0].replace(/\//g, '');
if (path === 'disk') return 'home';
if (path === 's' || path === 'share') return 'share';
return '';
},
showDownloadLog(fileList, fileStat) {
let theFile = fileList[0];
globalData.theFile = theFile;
let content = `
<div class="alert-primary" role="alert">
<div id="logTop">
<h5 class="alert-heading">正在获取 ${cutString(theFile.server_filename, 40)} 下载链接</h5>
</div>
<hr>
<div id="logleft">
<div id="logmeg">
<p class="card-text">下载地址8小时有效,请及时下载</p>
<p class="card-text">文件大小:<span id="fileMessgeSize"><b>正在获取...</b></span></p>
<p class="card-text">上传时间:<span id="fileMessgeTime"><b>正在获取...</b></span></p>
</div>
<hr>
<div id="logDownload">
<p class="card-text"><b style="color: #ff1a5a;">* 链接 ↓ ↓ ↓ :</b></p>
<p id="logUrl" class="card-text">
</p>
<p class="card-text">
<a id="logaria2" href="javascript:void(0);" >
发送到 Aria2(Motrix)
</a>
</p>
<p class="card-text"><a href="${globalData.domainB}?help" target="_blank">下载链接使用帮助(必读)</a></p>
</div>
<hr>
<div id="logTips"></div>
</div>
<div id="logright">
<div id="logdownload">
<div id="logCodeTips"></div>
<div id="logCodediv">
<div id="logCodeInput">
<span id="logCodeTips1"></span>
<input id="logCode" type="text" value="${getStorage('code')}" />
</div>
<input id="logdownloadBtn" type="button" value="点击获取直链" />
<div id="logCodeRemark">
</div>
</div>
</div>
<div id="logPi">
<img id="logPiImg" src="https://lingnan.minherd.top/attachment/2021/06/boge杂货铺.png" />
</div>
</div>
</div>
<hr style="clear: both; margin-top: 10px">
`;
this.Swal_log(content, '', 2, '', false)
.then(value => {
if (!value) {
window.open("https://greasyfork.org/zh-CN/scripts/428909/feedback", '_blank');
}
});
//绑定按钮点击(点击获取直链地址)
$("#logdownloadBtn").click(function () {
$("#logPi").css('visibility', 'hidden');
downloadClick()
});
/*$("#logaria2").click(function () {
main.Swal_log('暂不支持发送到Aria2,后续更新将开放')
});*/
},
};
//css
GM_addStyle(`
.swal-modal {
min-width: 740px;
width: auto;
}
.swal-footer{
clear: both;
margin-top: 5px;
}
.card-text {
font-size: 14px;
}
.alert-primary {
text-align: left;
margin: 0;
line-height: 2.12rem;
}
#logTop {
text-align: center;
}
#logleft {
float: left;
width: 50%;
margin-bottom: 10px;
}
#logright {
float: left;
width: 46%;
margin-left: 15px;
}
#logmeg,#logDownload,#logTips {
padding: 0 10px;
}
#logPi{
width: 265px;
height: 265px;
margin: 10px 0;
padding-left: 50%;
visibility: hidden;
}
#logPi img{
width: 100%;
left: -132.5px;
position: relative;
}
#logdownload {
text-align: center;
}
#logdownloadBtn {
width: 260px;
height: 40px;
background: #ffa700 !important;
border-radius: 4px;
transition: .3s;
font-size: 20px !important;
border: 0;
color: #fff;
cursor: pointer;
text-decoration: none;
font-family: Microsoft YaHei,SimHei,Tahoma;
font-weight: 100;
letter-spacing: 2px;
}
#logCodeRemark{
/*display: none;*/
text-align: left;
line-height: 1.5rem;
left: 50%;
position: relative;
}
#logCodeRemark p{
/*margin: 0 25px;*/
font-size: 12px;
left: -130px;
position: relative;
}
#logCodediv {
font-size: 12px;
border: 2px solid #EDD;
}
#logCodeTips{
display: none;
background: #ebebe8;
padding: 3px 14px;
color: #000000;
border-radius: 2px;
font-weight: bold;
text-align: left;
margin-top: 2px;
}
#logCodeTips1{
font-size: 16px;
}
#logCodeInput{
margin: 6px 0;
}
#logaria2 {
/*visibility: hidden;*/
display: none;
}
.point-success {
background-color: #28a745;
}
.point-lg {
width: 12px;
height: 12px;
}
.point {
display: inline-block;
//width: 5px;
//height: 5px;
border-radius: 500px;
margin: 0px 5px;
background-color: #ddd;
vertical-align: baseline;
`);
const DEBUG = false;
const divLogContent = document.createElement('div');
divLogContent.id = "divLogContent";
main.init();
})();