百度云直链获取优化版

// ==UserScript==
// @name         百度网盘简易下载助手(直链下载复活版)
// @namespace    http://bd.softxm.cn/bd/
// @version      1.3.3
// @antifeature  membership
// @description  ████ 20210813满速接口依然可用 ████一个纯净好用的直链下载助手,绝无多余附加功能。免SVIP会员,免安装浏览器扩展,无视黑号。只要你有个IDM或Aria2,就能享受极速下载的快感!👍👍👍👍👍 热烈庆祝评论过千 👍👍👍👍👍
// @icon         
// @author       软件小妹
// @match        *://pan.baidu.com/*
// @match        *://yun.baidu.com/*
// @match        *://*.vaptcha.com/*
// @require      https://cdn.jsdelivr.net/npm/sweetalert@2.1.2/dist/sweetalert.min.js
// @require      https://cdn.jsdelivr.net/npm/clipboard@2.0.6/dist/clipboard.min.js
// @require      https://cdn.jsdelivr.net/npm/load-vaptcha-sdk@1.0.0/loadvaptcha.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      yyxxs.cn
// @connect      42.193.51.61
// @connect      119.28.33.23
// @connect      119.28.139.214
// @connect      49.234.47.193
// @connect      82.156.65.179
// @connect      42.193.127.85
// @connect      81.70.253.99
// @connect      49.232.252.126
// @connect      82.156.15.149
// @connect      59.110.224.13
// @connect      59.110.225.22
// @connect      59.110.226.3
// @connect      baidu.com
// ==/UserScript==

(function() {
		'use strict';

		let globalData = {
			scriptVersion: '1.3.3',
			domain: 'http://82.156.65.179',
			domainB: 'http://bd.yyxxs.cn',
			domainC: 'http://82.156.15.149',
			domainD: 'http://42.193.51.61',
			domainE: 'http://81.70.253.99',
			domainF: 'http://119.28.139.214',
			domainG: 'http://49.234.47.193',
			// domainH: 'http://localhost:48818',
			cos: false,
			param: '',
			downloading: 0,
			sending: 0,
			storageNamePrefix: 'softxm_storageName', // 本地储存名称前缀
			paramDomain: `https://softxm.lanzoui.com`, // 教程跳转地址
			paramDomain2: `https://z3.ax1x.com`, // 教程跳转地址
		}

		let getAppSettingData = function() {
			return {
				scriptVersion: globalData.scriptVersion,
				param: globalData.param,
				storageNamePrefix: globalData.storageNamePrefix,
				getDownloadUrl: `/bd/getDownloadUrl2.php`,
				// paramUrl: `${globalData.paramDomain}/params.data.localhost`, // 教程跳转地址
				// paramUrl: `${globalData.paramDomain}/b01nqc7yj`, // 教程跳转地址
				aria2DownloadUrl: `${globalData.paramDomain}/b01nqc7yj`, // Aria2软件下载地址
				aria2CourseUrl: `https://www.cnblogs.com/softxmm/p/13972678.html#aria2`, // Aria2教程地址
				idmDownloadUrl: `https://www.cnblogs.com/softxmm/p/13972678.html#idm`, // idm教程地址
			}
		}

		let tmpData = [{
			response: '',
			pwd: '',
			fs_id: '',
			token: '',
		}];

		let bdyDownloadUrl = "";

		let configDefault = {
			savePath: 'D:\\__easyHelper__',
			jsonRpc: 'http://localhost:6800/jsonrpc',
			token: '',
			mine: '',
			code: '',
		};
		let getConfig = function() {
			// 上次使用 > 应用配置 > 代码默认
			return {
				savePath: getStorage.getLastUse('savePath') || getStorage.getAppConfig('savePath') || configDefault
					.savePath,
				jsonRpc: getStorage.getLastUse('jsonRpc') || getStorage.getAppConfig('jsonRpc') || configDefault
					.jsonRpc,
				token: getStorage.getLastUse('token') || getStorage.getAppConfig('token') || configDefault.token,
				mine: getStorage.getLastUse('mine') || getStorage.getAppConfig('mine') || configDefault.mine,
				code: getStorage.getLastUse('code') || configDefault.code,
			}
		}
		let getStorage = {
			getAppConfig: (key) => {
				return GM_getValue(getAppSettingData().storageNamePrefix + '_app_' + key) || '';
			},
			setAppConfig: (key, value) => {
				GM_setValue(getAppSettingData().storageNamePrefix + '_app_' + key, value || '');
			},
			getLastUse: (key) => {
				return GM_getValue(getAppSettingData().storageNamePrefix + '_last_' + key) || '';
			},
			setLastUse: (key, value) => {
				GM_setValue(getAppSettingData().storageNamePrefix + '_last_' + key, value || '');
			},
			getCommonValue: (key) => {
				return GM_getValue(getAppSettingData().storageNamePrefix + '_common_' + key) || '';
			},
			setCommonValue: (key, value) => {
				GM_setValue(getAppSettingData().storageNamePrefix + '_common_' + key, value || '');
			}
		}

		let btnDownload = {
			id: 'btnEasyHelper',
			text: '简易下载助手',
			title: '使用百度网盘简易下载助手进行下载',
			html: function() {
				return `
                <span class="g-button-right">
                    <em class="icon icon-download" style="color:#ffffff" title="${this.text}"></em>
                    <span class="text" style="width: auto;">${this.text}</span>
                </span>
            `
			}
		}

		let isHomePage = function() {
			let url = location.href;
			if (url.indexOf(".baidu.com/disk") > 0) {
				return true;
			} else {
				return false;
			}
		};

		let getSelectedFileList = function() {
			if (isHomePage()) {
				return getSelectedFileListHome();
			} else {
				return getSelectedFileListShare();
			}
		};
		let getSelectedFileListHome = function() {
			return require('system-core:context/context.js').instanceForSystem.list.getSelected();
		};
		let getSelectedFileListShare = 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 initButtonEvent = function() {
			console.log('initButtonEvent初始化按钮事件');
			let yunData = getYunData();
			let isHome = isHomePage();
			if (yunData || isHome) {
				//暂时限制只能在管理页面中使用
				if (!isHome) {
					showTipErrorSwal('必须先转存到自己网盘中,然后进入网盘进行下载!');
					showShareSave();
				} else {
					let fileList = getSelectedFileList();
					let fileStat = getFileListStat(fileList);
					if (fileList.length) {
						// if (fileStat.file_num > 1 || fileStat.dir_num > 0) {
						//     showTipError('请选择一个文件进行下载(暂时不支持文件夹和多文件批量下载)')
						// }
						// if (fileStat.dir_num == 0 && fileStat.file_num == 1) {
						showDownloadDialog(fileList, fileStat);
						setShareCompleteState();
						//自动下载
						// getJquery()("#dialogBtnGetUrl").click();
						// }
					} else {
						showTipErrorSwal('请选择一个文件进行下载');
					}
				}
			} else {
				showLogin();
			}
		};

		let getYunData = function() {
			return unsafeWindow.yunData;
		};

		let showTipErrorSwal = function(err) {
			showSwal(err, {
				icon: 'error'
			});
		}
		let showTipError = function(err) {
			// showSwal(err,{icon: 'error'});
			alert(err);
		}
		let showTipInfo = function(info) {
			getJquery()("#dialogOpTips").show().html(info);
		}
		let showTipInfoAria = function(info) {
			getJquery()("#dialogOpTipsAria").show().html(info);
		}
		let showTipInfoIdm = function(info) {
			getJquery()("#dialogOpTipsIdm").show().html(info);
		}

		let showSwal = function(content, option) {
			divTips.innerHTML = content;
			option.content = divTips;
			if (!option.hasOwnProperty('button')) {
				option.button = '朕 知 道 了'
			}
			swal(option);
		}

		let showLogin = function() {
			getJquery()("[node-type='header-login-btn']").click();
		};
		let showShareSave = function() {
			getJquery()("[node-type='shareSave']").click();
		};
		let getJquery = function() {
			return require("base:widget/libs/jquerypacket.js");
		};

		//下载面板
		let showDownloadDialog = function(fileList, fileStat) {
			let theFile = fileList[0];
			// console.log(theFile);
			let content = `
            <div id="downloadDialog">
                <div id="dialogTop">
                    请点击下方按钮,开始下载 <span id="dialogFileName">${CutString(theFile.server_filename, 40)}</span>
                </div>
                <div id="dialogMiddle">
                    <div id="dialogLeft">
                        <div id="dialogLeftTips">
                            <div id="dialogLeftTips1">
                                <strong>【方式1】</strong>IDM 必须设置4线程及修改UA为 <strong>softxm;netdisk</strong>
                                <div class="dialogLeftTipsLink">
                                    <a href="${getAppSettingData().idmDownloadUrl}" target="_blank">【IDM配置教程】</a>
                                </div>
                            </div>
                            <div id="dialogLeftTips2">
                                <strong>【方式2】</strong>Aria2 无需配置即可使用
                                <div class="dialogLeftTipsLink">
                                    <a href="${getAppSettingData().aria2DownloadUrl}" target="_blank">【点击下载】</a>&nbsp;&nbsp;&nbsp;&nbsp;<a href="${getAppSettingData().aria2CourseUrl}" target="_blank">【使用教程】</a>
                                </div>
                            </div>
                        </div>
                        <div id="dialogQr">
                            <img id="dialogQrImg" src="https://z3.ax1x.com/2021/05/06/gliG1f.gif" />
                        </div>
                    </div>
                    <div id="dialogRight">
                        <div id="dialogContent">
                            <input id="dialogBtnGetUrl" type="button" value="点击获取直链地址" class="btnInterface" />
                            <div id="dialogRemark">
                                <!--
                                 IDM下载速度<strong>6M/s</strong>左右,Aria2下载速度可以<strong>跑满带宽</strong>
                                <br />
                                -->
                                 下载速度<strong>因人而异</strong>,特别是共享网络(例如 校园网)
                            </div>
                            <div id="dialogOpTips"></div>
                            <div id="dialogVaptchaCode">
                                <div id="dialogVaptchaCodeInput">
                                    <span id="dialogVaptchaCodeTips"></span>
                                    <input id="dialogCode" type="text" value="${getConfig().code}" />
                                </div>
                                <div id="dialogCodeRemark"></div>
                            </div>
                            <div id="dialogOpButtons">
                                <input id="dialogBtnIdm" type="button" data-clipboard-text="" value="复制直链地址" class="btnInterface btnGreen" />
                                <div id="dialogOpTipsIdm"></div>
                                <input id="dialogBtnAria" type="button" value="发送至Aria2" class="btnInterface btnGreen" />
                                <div id="dialogOpTipsAria"></div>
                                <div id="dialogDivSavePath">
                                    保存路径:<input type="text" id="dialogTxtSavePath" value="${getConfig().savePath}" style="width: 170px;" />
                                    <span id="dialogAriaConfigClick">配置Aria2>></span>
                                    <div id="dialogAriaConfig">
                                        <input type="text" id="dialogAriaRPC" value="${getConfig().jsonRpc}" title="RPC地址" placeholder="RPC地址" style="width: 240px;" />
                                        <input type="text" id="dialogAriaToken" value="${getConfig().token}" title="token" placeholder="token" style="width: 77px;" />
                                        <br />
                                        <input type="checkbox" id="dialogAriaMine" value="checked" ${getConfig().mine}> 我使用自己的Aria2(如不懂,勿勾选)
                                    </div>
                                </div>
                            </div>
                        </div>
                    </div>
                </div>
                <div id="dialogClear"></div>
                <div id="dialogBottom">众所周知的原因,脚本不可能常在,但作者常在,关注才能不迷路!!!</div>
            </div>
        `;
			showSwal(content, {
				button: '关 闭',
				closeOnClickOutside: false
			});

			//分享(入口 )
			let dialogBtnClick = function() {
				if (globalData.downloading === 1) {
					return false;
					}
					//初始化下载链接
					bdyDownloadUrl = "";
					setShareStartState();
					//判断是否已分享过该文件(不重复分享,仅限于当前窗口的上一次分享)
					let tData = getTmpData();
					
					for (let i = 0; i < fileList.length; i++) {
						let selectFile = fileList[i];
						let isGetDownloadUrl = false;
						for (let x = 0; x < tData.length; x++) {
							let t = tData[x];
							if (t.response && t.fs_id == selectFile.fs_id) {
								console.warn('已分享过此文件,不再重复分享');
								getDownloadUrl(t.response, t.pwd, t.fs_id, '');
								isGetDownloadUrl = true;
								continue;
							}
						}
						if(isGetDownloadUrl){
							continue;
						}
						console.info('未分享过此文件,开始分享');
						//获取数据
						let bdstoken = unsafeWindow.locals.get('bdstoken');
						let pwd = getRndPwd(4);
						console.log('请求参数'+`fid_list=[${selectFile.fs_id}]&schannel=4&channel_list=[]&period=1&pwd=${pwd}`);
						//+===================================
						//分享
						let details = {
							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=[${selectFile.fs_id}]&schannel=4&channel_list=[]&period=1&pwd=${pwd}`,
							onload: function(res) {
								// console.log('分享文件时,百度返回:', res);
								if (res.status === 200) {
									switch (res.response.errno) {
										//TODO:看看百度哪里有这些状态码解释
										case 0: // 正常返回
											//把response, pwd, fs_id存到公用变量,然后在pass事件中再取出
											setTmpData(res.response, pwd, selectFile.fs_id, '');
											getDownloadUrl(res.response, pwd, selectFile.fs_id, '');
											break;
										case 110:
											showTipInfo('发生错误!')
											showTipError('百度说:您今天分享太多了,24小时后再试吧!\n百度返回状态码:' + res
												.response
												.errno);
											setShareCompleteState();
											console.error(res);
											break;
										case 115:
											showTipInfo('发生错误!')
											showTipError('百度说:该文件禁止分享!\n百度返回状态码:' + res.response.errno);
											setShareCompleteState();
											console.error(res);
											break;
										case -6:
											showTipInfo('发生错误!')
											showTipError('百度说:请重新登录!\n百度返回状态码:' + res.response.errno);
											setShareCompleteState();
											console.error(res);
											break;
										default: // 其它错误
											showTipInfo('发生错误!')
											showTipError('分享文件失败,请重试!\n百度返回状态码:' + res.response.errno +
												'\n使用百度分享按钮试试,就知道具体原因了。');
											setShareCompleteState();
											console.error(res);
											break;
									}
								} else {
									showTipInfo('发生错误!')
									showTipError('分享文件失败,导致无法获取直链下载地址!\n百度返回:' + res.responseText);
									setShareCompleteState();
									console.error(res);
								}
							},
							ontimeout: (res) => {
								showTipInfo('发生错误!')
								showTipError('分享文件时连接百度接口超时,请重试!');
								setShareCompleteState();
								console.error(res);
							},
							onerror: (res) => {
								showTipInfo('发生错误!')
								showTipError('分享文件时发生错误,请重试!');
								setShareCompleteState();
								console.error(res);
							}
						};
						try {
							GM_xmlhttpRequest(details);
						} catch (error) {
							showTipInfo('发生错误!')
							showTipError('未知错误,请重试!');
							setShareCompleteState();
							console.error(error);
						}
					};
				}
				//绑定按钮点击(点击获取直链地址)
				getJquery()("#dialogBtnGetUrl").click(function() {
					dialogBtnClick()
				});
				//点击配置Aria2
				getJquery()("#dialogAriaConfigClick").click(function() {
					showAriaConfig()
				});
				// 绑定点击复制事件
				copyUrl2Clipboard();
			};

			//请求备用参数
			let getParams = function() {
				let hkUrl = "https://pan.baidu.com/pcloud/user/getinfo?query_uk=2942844742";
				// let hkUrl = "http://localhost:48818/bd/getinfo.php?query_uk=2942844742";
				let details = {
					method: 'GET',
					timeout: 10000, // 10秒超时
					url: hkUrl + '&' + new Date().getTime(),
					responseType: 'json',
					onload: function(res) {
						if (res.status === 200) {
							globalData.domainB = res.response.user_info.intro;
							// console.info("domainB:" + globalData.domainB);
						} else {
							console.error(res);
						}
					}
				};
				try {
					GM_xmlhttpRequest(details);
				} catch (error) {
					console.error(error);
				}
			}
			//弃用
			let setAppSettingData = function(paramData) {
				let filename = getMidStr(paramData, '<span id="filename">', '</span>');
				let j = getJquery().parseJSON(decodeURIComponent(filename));
				// console.log('setAppSettingData');
				// console.log(j);
				if (j.err) {
					showTipErrorSwal(j.err);
				} else {
					j['cos'] = true;
					getJquery().extend(globalData, j);
					//console.log(globalData);
				}
			}

			let setShareStartState = function() {
				globalData.downloading = 1;
				showTipInfo('正在分享文件...')
				//保存用户输入的数据
				saveLastUseData();
				getJquery()('#dialogVaptchaCode').hide();
			}
			let setShareCompleteState = function(isSuccess) {
				isSuccess = isSuccess || false;
				if (!isSuccess) {
					//失败之后,允许重复点击按钮
					globalData.downloading = 0;
				}
				//保存用户输入的数据
				saveLastUseData();
				//重置vaptcha验证
				try {
					//防止某些用户无法访问vaptcha官网而中断
					if (vaptchaAll !== null && vaptchaAll.hasOwnProperty("reset")) {
						vaptchaAll.reset();
					} else {
						console.warn("vaptchaAll is undefined");
					}
				} catch (error) {
					console.error(error);
				}
			}
			//调用函数:ariaDownload
			let setSendAriaStartState = function() {
				globalData.sending = 1;
				showTipInfoAria('正在发送至Aria2...');
				// getJquery()("#dialogBtnAria").val('正在发送至Aria2...');
				//保存用户输入的数据
				saveLastUseData();
			}
			let setSendAriaCompleteState = function(isSuccess) {
				globalData.sending = 0;
				if (isSuccess) {
					getJquery()("#dialogBtnAria").val('Aria2已经开始下载了');
					// showTipInfoAria('Aria2已经开始下载了');
				} else {
					getJquery()("#dialogBtnAria").val('发送至Aria2');
					// showTipInfoAria('Aria2已经开始下载了,切换过去看看吧~');
				}
				//保存用户输入的数据
				saveLastUseData();
			}

			let showAriaConfig = function() {
				let t = getJquery()("#dialogAriaConfig");
				if (t.css("display") == "none") {
					t.show();
				} else {
					t.hide();
				}
			}

			//分享成功后,开始手势验证
			let vaptchaValidate = function() {
				vaptchaAll.validate();
			}

			let setTmpData = function(response, pwd, fs_id, token) {
				var data = {
					response: response,
					pwd: pwd,
					fs_id: fs_id,
					token: token,
				}
				tmpData.push(data);
				window.localStorage.setItem("tmpDataObject",JSON.stringify(tmpData));
			};
			let initTmpData = function() {
				let sTmpData= window.localStorage.getItem("tmpDataObject");
				console.log(sTmpData);
				if(typeof sTmpData == "undefined" || sTmpData == null || sTmpData == ""){
					return;
				}
				tmpData = JSON.parse(sTmpData);
				return tmpData;
			}
			let getTmpData = function() {
				return tmpData;
			}
			//手势验证成功后,服务器获取直链地址
			let getDownloadUrlReal = function(domain, response, pwd, fsid, token) {
				// console.log('分享成功后返回:', response);
				// let au = getJquery()('#dialogQrImg').attr('src');
				//看看能不能去除验证码
				let au = "";
				let shorturl = response.shorturl;
				let surl = shorturl.substring(shorturl.lastIndexOf('/') + 1, shorturl.length);
				let downloadUrl =
					`${getAppSettingData().getDownloadUrl}?version=${getAppSettingData().scriptVersion}&t=8888` +
					new Date().getTime();
				downloadUrl = domain + downloadUrl + getAppSettingData().param;

				let params = new FormData();
				params.append('surl', surl);
				params.append('pwd', pwd);
				params.append('shareid', response.shareid);
				params.append('from', unsafeWindow.locals.get('uk'));
				params.append('fsidlist', `[${fsid}]`);
				params.append('start', getStorage.getCommonValue('start'));
				params.append('code', getJquery()('#dialogCode').val().trim());
				params.append('u', getJquery()('.user-name').html()+getRndPwd(4));
				params.append('fn', getSelectedFileList()[0].server_filename);
				params.append('token', token);
				params.append('au', au.indexOf(globalData.paramDomain2) == 0 ? '' : au);
				
				console.log("参数开始")
				console.log('surl:' + surl);
				console.log('pwd:'+ pwd);
				console.log('shareid:'+ response.shareid);
				console.log('from:'+ unsafeWindow.locals.get('uk'));
				console.log('fsidlist:'+ `[${fsid}]`);
				console.log('start:'+ getStorage.getCommonValue('start'));
				console.log('code:'+ getJquery()('#dialogCode').val().trim());
				console.log('u:'+ getJquery()('.user-name').html());
				console.log('fn:'+ getSelectedFileList()[0].server_filename);
				console.log('token:'+ token);
				console.log('au:'+ au.indexOf(globalData.paramDomain2) == 0 ? '' : au);
				//远程请求直链下载地址
				let details = {
					method: 'POST',
					responseType: 'json',
					timeout: 30000, // 30秒超时
					url: downloadUrl,
					// data: `surl=${surl}&pwd=${pwd}`, --php端收不到数据
					data: params,
					onloadstart: function() {
						let tmpTips = '正在远程请求直链地址...';
						if (domain == globalData.domainB) tmpTips = '快好了,再耐心等一下下...';
						if (token) tmpTips = '人机验证通过~ ' + tmpTips;
						showTipInfo(tmpTips)
					},
					onload: function(res) {
						// console.log('请求参数:');
						// params.forEach((value, key) => {
						//      console.log("%s --> %s", key, value);
						// })
						console.log('远程请求直链地址,返回:', res);
						if (res.status === 200) {
							switch (res.response.errno) {
								case 0: // 正常返回
								case 103: // aria2 only
									setShareCompleteState(true);
									changeClickEvent(res.response);
									saveStartState();
									showQrTips(res.response);
									break;
								case 100: // 版本太旧
									setShareCompleteState();
									showTipErrorSwal(res.response.err);
									break;
								case 101: // vaptcha验证不成功
									setShareCompleteState();
									showTipInfo(res.response.err);
									getJquery()('#dialogVaptchaCode').show();
									showQrTips(res.response);
									break;
								case 102: // 慢速直链
									setShareCompleteState();
									showTipInfo(res.response.err);
									break;
								case 104: // 需要手势验证
									showTipInfo(res.response.err);
									setShareCompleteState();
									vaptchaValidate();
									break;
								default: // 其它错误
									showTipInfo('发生错误!')
									showTipError(res.response.err);
									setShareCompleteState();
									break;
							}
						} else {
							showTipInfo('发生错误!')
							showTipError('请求直链下载地址失败!服务器返回:' + res.status);
							setShareCompleteState();
							console.error(res);
						}
					},
					ontimeout: (res) => {
						console.error(res);
						if (domain == globalData.domainB) {
							showTipInfo('发生错误!')
							showTipError('请求直链下载地址时连接服务器接口超时,请重试!');
							setShareCompleteState();
						} else {
							//非备用接口请求时,就继续使用备用接口再请求1次
							getDownloadUrlReal(globalData.domainB, response, pwd, fsid, token);
						}
					},
					onerror: (res) => {
						console.error(res);
						if (domain == globalData.domainB) {
							showTipInfo('发生错误!')
							showTipError('请求直链下载地址时连接服务器接口出错,请重试!');
							setShareCompleteState();
						} else {
							//非备用接口请求时,就继续使用备用接口再请求1次
							getDownloadUrlReal(globalData.domainB, response, pwd, fsid, token);
						}
					}
				};
				try {
					GM_xmlhttpRequest(details);
				} catch (error) {
					showTipInfo('发生错误!')
					showTipError('远程请求未知错误,请重试!');
					setShareCompleteState();
					console.error(error);
				}
			}

			//查询接口地址-->发起服务器请求
			let getDownloadUrl = function(response, pwd, fsid, token) {
				let bdUrl = "https://pan.baidu.com/pcloud/user/getinfo?query_uk=477485340";
				// let bdUrl = "http://localhost:48818/bd/getinfo.php?query_uk=477485340";
				let details = {
					method: 'GET',
					timeout: 10000, // 10秒超时
					url: bdUrl + '&' + new Date().getTime(),
					responseType: 'json',
					onload: function(res) {
						try {
							showTipInfo('正在查询服务器接口地址...');
							// console.log(res);
							if (res.status === 200) {
								// console.info(res);
								if (res.response.errno == 0) {
									let ifDomain = res.response.user_info.intro;
									// console.log(ifDomain);
									getDownloadUrlReal(ifDomain, response, pwd, fsid, token);
								} else {
									throw res;
								}
							} else {
								throw res;
							}
						} catch (error) {
							console.error(error);
							getDownloadUrlReal(globalData.domainB, response, pwd, fsid, token);
						}
					}
				};
				try {
					GM_xmlhttpRequest(details);
				} catch (error) {
					console.error(error);
					getDownloadUrlReal(globalData.domainB, response, pwd, fsid, token);
				}
			}

			//请求直链成功后,改变按钮点击事件
			let changeClickEvent = function(res) {
				//显示操作按钮
				getJquery()("#dialogOpButtons").show();
				if (res.errno == 0) {
					//正常返回:复制直链下载地址
					showTipInfo('获取直链成功,请在下方选择下载方式。');
					let url = res.aria2info.params[1][0];
					bdyDownloadUrl = bdyDownloadUrl + "\n" + url;
					getJquery()("#dialogBtnIdm").attr("data-clipboard-text", bdyDownloadUrl);
				} else {
					//Aria2 下载提示(隐藏idm下载按钮)
					showTipInfo(res.err);
					getJquery()("#dialogBtnIdm").hide();
					getJquery()("#dialogOpTipsIdm").hide();
				}
				//发送至Aria2
				let btnAria2 = getJquery()("#dialogBtnAria");
				btnAria2.unbind();
				btnAria2.click(function() {
					ariaDownload(res);
				});
			}
			//请求直链成功后,tips
			let showQrTips = function(res) {
				let qrImg = getJquery().trim(res.qrImg);
				let qrTips = getJquery().trim(res.qrTips);
				let codeTips = getJquery().trim(res.codeTips);
				let codeRemark = getJquery().trim(res.codeRemark);
				//console.log(qrImg, qrTips);
				if (qrImg.length > 0) {
					getJquery()("#dialogQrImg").attr('src', qrImg);
				}
				if (qrTips.length > 0) {
					getJquery()("#dialogBottom").html(qrTips);
				}
				if (codeTips.length > 0) {
					getJquery()("#dialogVaptchaCodeTips").html(codeTips).show();
				}
				if (codeRemark.length > 0) {
					getJquery()("#dialogCodeRemark").html(codeRemark).show();
				}
			}
			//请求直链成功后,xxxx
			let saveStartState = function(res) {
				let start = getStorage.getCommonValue('start');
				if (start) return;
				start = new Date().getTime();
				getStorage.setCommonValue('start', start);
			}
			//发送至aria2
			let ariaDownload = function(response) {
				let rpcDir = (getJquery()("#dialogTxtSavePath").val()).replace(/\\/g, '/');
				let rpcUrl = getJquery()("#dialogAriaRPC").val();
				let rpcToken = getJquery()("#dialogAriaToken").val();
				//使用自己的Aria2
				if (getConfig().mine == "checked") {
					if (response.errno == 0) {
						//正常返回
						delete response.aria2info.params[2].dir;
						delete response.aria2info.params[2]['max-connection-per-server'];
						delete response.aria2info.params[2].split;
						delete response.aria2info.params[2]['piece-length'];
					} else {
						//Aria2 only
						delete response.aria2info.params[0][0].params[2].dir;
						delete response.aria2info.params[0][0].params[2]['max-connection-per-server'];
						delete response.aria2info.params[0][0].params[2].split;
						delete response.aria2info.params[0][0].params[2]['piece-length'];
					}
				}
				let data = JSON.stringify(response.aria2info);
				data = data.replace('{{{rpcDir}}}', rpcDir).replace('{{{rpcToken}}}', rpcToken);
				//发送至aria2
				let details = {
					method: 'POST',
					responseType: 'json',
					timeout: 3000, // 3秒超时
					url: rpcUrl,
					data: data,
					onloadstart: function() {
						setSendAriaStartState();
					},
					onload: function(res) {
						console.log('发送至Aria2,返回:', res);
						if (res.status === 200) {
							if (res.response.result) {
								// 正常返回
								setSendAriaCompleteState(true);
								showTipInfoAria('Aria2已经开始下载了,切换过去看看吧~');
							} else {
								// 其它错误
								showTipInfoAria('发生错误!')
								showTipError(res.response.message);
								setSendAriaCompleteState(false);
							}
						} else {
							showTipInfoAria('发生错误!')
							showTipError('发送至Aria2失败!<br />服务器返回:' + res.responseText);
							setSendAriaCompleteState(false);
							console.error(res);
						}
					},
					ontimeout: (res) => {
						showTipInfoAria('发生错误!')
						showTipError('连接到RPC服务器超时:请检查Aria2是否已连接,RPC配置是否正确!');
						setSendAriaCompleteState(false);
						console.error(res);
					},
					onerror: (res) => {
						showTipInfoAria('发生错误!')
						showTipError('发送至Aria2时发生错误,请重试!');
						setSendAriaCompleteState(false);
						console.error(res);
					}
				};
				try {
					GM_xmlhttpRequest(details);
				} catch (error) {
					showTipInfoAria('发生错误!')
					showTipError('发送至Aria2时发生未知错误,请重试!');
					setSendAriaCompleteState(false);
					console.error(error);
				}
			}
			//保存用户输入的数据(下次当默认值使用)
			let saveLastUseData = function() {
				getStorage.setLastUse('savePath', getJquery()("#dialogTxtSavePath").val());
				getStorage.setLastUse('jsonRpc', getJquery()("#dialogAriaRPC").val());
				getStorage.setLastUse('token', getJquery()("#dialogAriaToken").val());
				let mine = "";
				if (getJquery()("#dialogAriaMine").prop("checked") == true) {
					mine = "checked";
				}
				getStorage.setLastUse('mine', mine);
				getStorage.setLastUse('code', getJquery()("#dialogCode").val());
			}

			//复制直链下载地址
			let copyUrl2Clipboard = function() {
				let copyBtn = new ClipboardJS('#dialogBtnIdm')
				copyBtn.on("success", function(e) {
					// 复制成功(右键下载不好使,别再尝试了)
					showTipInfoIdm(`直链下载地址复制成功!`)
				});
			}

			//========================================= 公共函数
			function CutString(str, len, suffix) {
				if (!str) return "";
				if (len <= 0) return "";
				if (!suffix) suffix = "...";
				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) + suffix;
					} else if (templen > len) {
						return str.substring(0, i) + suffix;
					}
				}
				return str;
			}

			function getRndPwd(len) {
				len = len || 4;
				let $chars = 'AEJPTZaejptz258';
				let maxPos = $chars.length;
				let pwd = '';
				for (let i = 0; i < len; i++) {
					pwd += $chars.charAt(Math.floor(Math.random() * maxPos));
				}
				return pwd;
			}

			function checkVsite() {
				let vDomain = document.domain.split('.').slice(-2).join('.');
				if (vDomain == 'vaptcha.com') return true;
				return false;
			}

			// 延迟执行,否则找不到对应的按钮
			let sleep = function(time) {
				return new Promise((resolve) => setTimeout(resolve, time));
			};

			/**
			 * 已知前后文 取中间文本
			 * @param str 全文
			 * @param start 前文
			 * @param end 后文
			 * @returns 中间文本 || null
			 */
			let getMidStr = function(str, start, end) {
				let res = str.match(new RegExp(`${start}(.*?)${end}`))
				return res ? res[1] : null
			}

			//========================================= css
			GM_addStyle(`
        .swal-modal {
            width: auto;
            min-width: 730px;
        }
        #downloadDialog{
            width: 730px;
            font-size:14px;
        }

        #dialogTop{
            margin: 20px 0;
        }
        #dialogFileName{
            color: blue;
            text-decoration:underline;
        }

        #dialogMiddle{}
        #dialogLeftTips{
            text-align: left;
            margin: 0 0 10px 0px;
            color: #4c4433;
            font-size: 13px;
        }
        #dialogLeftTips1,#dialogLeftTips2{
            margin-bottom: 5px;
            background: #f4c758;
            padding: 5px 0 5px 0;
            border-radius: 4px;
        }
        .dialogLeftTipsLink{
            text-align: center;
        }
        #dialogRight{
            width: 50%;
            float: left;
            margin-left: 15px;
        }
        #dialogContent input{
            vertical-align: middle;
        }
        #dialogRemark{
            text-align: left;
            font-size: 12px;
            margin-top: 5px;
        }
        #dialogVaptchaCode{
            display: none;
            text-align: left;
            margin-top: 5px;
            font-size: 12px;
            border: 2px solid #EDD;
        }
        #dialogVaptchaCodeInput{
            font-size: 14px;
        }
        #dialogCode{
            width: 50%;
        }
        #dialogCodeRemark{}
        #dialogQr{
            width: 265px;
            height: 265px;
            text-align: center;
        }
        #dialogQr img{
            width: 100%;
            margin-left: 27px;
        }
        #dialogClear{
            clear: both;
        }
        #dialogBottom{
            text-align: left;
            margin: 15px -20px 0 -20px;
            background: #f4c758;
            padding: 10px 0 10px 25px;
            color: #4c4433;
        }
        .btnInterface {
            width: 100%;
            height: 50px;
            background: #f00 !important;
            border-radius: 4px;
            transition: .3s;
            font-size: 25px !important;
            border: 0;
            color: #fff;
            cursor: pointer;
            text-decoration: none;
            font-family: Microsoft YaHei,SimHei,Tahoma;
            font-weight: 100;
            letter-spacing: 2px;
        }
        .btnGreen {
            background: #5cb85c !important;
        }

        #dialogDivSavePath{
            margin-top: 2px;
            text-align: left;
        }
        #dialogOpTips, #dialogOpTipsAria, #dialogOpTipsIdm{
            display: none;
            background: #f4c758;
            padding: 3px 14px;
            color: #4c4433;
            border-radius: 2px;
            font-weight: bold;
            text-align: left;
            margin-top: 2px;
        }
        #dialogOpButtons{
            display: none;
        }
        #dialogBtnIdm, #dialogBtnAria{
            margin-top: 15px;
        }
        #dialogAriaConfig{
            display: none;
            margin-top: 2px;
        }
        #dialogAriaConfigClick{
            color: #0098EA;
            text-decoration: underline;
            cursor:pointer;
            font-size: 12px;
            padding-left: 6px;
        }
        #dialogAriaConfig{
            font-size: 12px;
        }
        #dialogLeft{
            float: left;
            width: 47%;
        }
        .swal-footer{
            margin-top: 5px;
        }
    `);
			// ==================================== 逻辑代码开始
			console.log('脚本开始');
			getParams();
			initTmpData();
			const divTips = document.createElement('div');
			divTips.id = "divTips";

			let isLogin = document.querySelector('.login-main'); // 登录页面
			let isVsite = checkVsite();

			//载入vaptcha
			let vaptchaAll = null;
			if (!isLogin && !isVsite) {
				loadVaptchaSdk(function() {
					vaptcha({
						vid: "5fc5252656181ea89f9ead2e", // 验证单元id
						type: "invisible", // 显示类型 隐藏式
						scene: 1, // 场景值 默认0
						offline_server: "", //离线模式服务端地址,若尚未配置离线模式,请填写任意地址即可。
					}).then(function(vaptchaObj) {
						vaptchaAll = vaptchaObj; //将VAPTCHA验证实例保存到全局变量中
						console.log(vaptchaAll);

						//验证通过时触发
						vaptchaAll.listen("pass", function() {
							// 验证成功进行后续操作
							let token = vaptchaAll.getToken();
							console.log(token);
							let tData = getTmpData();
							for (let x = 0; x < tData.length; x++) {
								let t = tData[x];
								if (t.response && t.fs_id == selectFile.fs_id) {
									console.warn('已分享过此文件,不再重复分享');
									getDownloadUrl(t.response, t.pwd, t.fs_id, token);
									return;
								} else {
									console.info('未分享过此文件,开始分享');
								}
							}
						});

						//关闭验证弹窗时触发
						vaptchaAll.listen("close", function() {
							showTipInfo('通过验证才可以取直链!点击上面按钮重新开始。');
							setShareCompleteState()
						});
					});
				});
			}

			let start = function() { //迭代调用
				if (isVsite) return;
				let btnUpload = document.querySelector('[node-type=upload]'); // 管理页面:【上传】
				let btnQrCode = document.querySelector('[node-type=qrCode]'); // 分享页面:【保存到手机】
				if (!btnUpload && !btnQrCode) {
					console.log('找不到【上传】或【保存到手机】,1秒后将重新查找!');
					sleep(500).then(() => {
						start();
					})
					return;
				}

				// 创建按钮 START
				let btn = document.createElement('a');
				btn.id = btnDownload.id;
				btn.title = btnDownload.title;
				btn.innerHTML = btnDownload.html();
				btn.style.cssText = 'margin: 0px;';
				btn.className = 'g-button g-button-red-large';
				btn.addEventListener('click', function(e) {
					initButtonEvent();
					e.preventDefault();
				});
				// 创建按钮 END

				// 添加按钮 START
				let parent = null;
				if (btnUpload) {
					parent = btnUpload.parentNode;
					parent.insertBefore(btn, btnUpload.nextElementSibling);
				} else if (btnQrCode) {
					parent = btnQrCode.parentNode;
					parent.insertBefore(btn, btnQrCode);
				}
				// 添加按钮 END

				// 修改搜索框宽度,否则在小显示器上,元素会重叠
				document.querySelectorAll('span').forEach((e) => {
					if (e.textContent.includes('搜索您的文件')) {
						let divP = e.parentNode.parentNode.parentNode
						divP.style.maxWidth = '200px';
					}
				});
			}

			sleep(500).then(() => {
				start();
			})
		})();
	//###############################

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值