Greasy Fork is available in English.

百度网盘免会员享svip下载

基于开放API,支持Windows,Mac,Linux等多平台,可使用IDM,Xdown等多线程加速工具加速下载直链下载助手了。免SVIP会员,免安装浏览器扩展。只要你有个Aria2或IDM,就可以使用你自己的帐号,享受极速下载的快感!

// ==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;">*  链接&nbsp;&nbsp;↓&nbsp;↓&nbsp;↓&nbsp;:</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();

})();