(改)网盘直链下载助手

可以获取网盘文件真实下载地址。基于【网盘直链下载助手】修改自5.9.4版本,自用,去推广,修原有BUG,修改界面。

当前为 2023-03-13 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name (改)网盘直链下载助手
  3. // @namespace https://github.com/syhyz1990/baiduyun
  4. // @version 1.0.5.4
  5. // @author Hmjz100、油小猴
  6. // @icon data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAxMjggMTI4Ij48cGF0aCBkPSJNMTAzLjYgMTA3LjRjMy41LTIuMiA4LjktNi4xIDEzLjgtMTIuNXM3LjMtMTIuNSA4LjUtMTYuNWMuNS0xLjcgMi4yLTcuNSAyLjItMTQuNyAwLTEwLjEtMy4zLTI1LjEtMTUuNC0zNi44LTE0LjUtMTQtMzIuMS0xNC4zLTM1LjctMTQuMy04IDAtMTUuNyAxLjktMjIuNiA1LjJDNDQgMjMgMzUuNyAzMS40IDMwLjggNDEuN2MtMS4zIDIuOC00IDQuNy03LjEgNS00IC4zLTcuNSA0LjQtOC45IDkuNi0uNSAxLjktMS42IDMuNS0zLjEgNC43QzQuNCA2Ni44IDAgNzUuNyAwIDg1YzAgNi44IDIuMyAxMy4xIDYuMSAxOC4yIDUuNSA3LjQgMTQuMiAxMi4yIDI0IDEyLjJoNDcuMWM0LjQgMCAxMS0uNSAxOC4zLTMuNSAzLjItMS40IDUuOS0zIDguMS00LjV6IiBmaWxsPSIjNDQ0Ii8+PHBhdGggZD0iTTExOS44IDY0LjNjLjEtMTcuMS0xMC40LTI4LTEyLjUtMzAuMUM5NSAyMi4xIDc5LjkgMjEuOCA3Ni45IDIxLjhjLTE3LjYgMC0zMy4zIDEwLjUtMzkuOSAyNi43LS42IDEuMy0xLjggMi4zLTMuNCAyLjNoLS40Yy01LjggMC0xMC42IDQuOC0xMC42IDEwLjd2LjVjMCAxLjQtLjggMi42LTEuOSAzLjNDMTMuNCA2OSA4LjggNzYuOCA4LjggODVjMCAxMi4yIDkuOSAyMi4zIDIyLjIgMjIuM2g0NS4yYzMuNi0uMSAxNy42LS45IDI5LjYtMTIgMi45LTIuOCAxMy45LTEzLjcgMTQtMzF6IiBmaWxsPSIjMTM5N2Q4Ii8+PHBhdGggZD0iTTExMC44IDU3LjRsLjIgMy4zYzAgMS4zLTEuMSAyLjQtMi4zIDIuNC0xLjMgMC0yLjMtMS4xLTIuMy0yLjRsLS4xLTIuOHYtLjNjMC0xLjIuOS0yLjIgMi4xLTIuM2guM2MuNyAwIDEuMy4zIDEuNy43LS4yLjEuMy41LjQgMS40em0tMy4zLTEwLjNjMCAxLjItMSAyLjMtMi4yIDIuM2gtLjFjLS44IDAtMS42LS41LTItMS4yLTQuNi04LjMtMTMuMy0xMy41LTIyLjgtMTMuNS0xLjIgMC0yLjMtMS0yLjMtMi4ydi0uMWMwLTEuMiAxLTIuMyAyLjItMi4zaC4xYTMwLjM3IDMwLjM3IDAgMCAxIDE1LjggNC40YzQuNiAyLjggOC40IDYuOCAxMS4xIDExLjUuMS4zLjIuNy4yIDEuMXpNODguMyA3My44TDczLjUgOTMuMmMtMS41IDEuOS0zLjUgMy4xLTUuNyAzLjVoLS4yYy0uNC4xLS44LjEtMS4yLjEtLjYgMC0xLjEtLjEtMS42LS4yLTIuMi0uNC00LjItMS43LTUuNi0zLjVMNDQuMyA3My45Yy0yLTIuNi0yLjUtNS40LTEuNC03LjcuMS0uMS4xLS4yLjItLjIgMS4yLTIgMy41LTMuMiA2LjQtMy4yaDYuNnYtNS43YzAtNi44IDQuNy0xMiAxMC45LTEyIDQuOCAwIDguNSAyLjYgMTAuMyA3LjIuNSAxLjMtLjIgMi43LTEuNSAzLjJzLTIuOC0uMS0zLjMtMS40Yy0xLjEtMi43LTIuOS00LTUuNS00LTMuNSAwLTYgMy02IDd2OC4xYzAgLjUtLjIgMS0uNiAxLjQtLjYuNy0xLjcgMS4xLTIuNiAxLjFoLTguNGMtMS4zIDAtMiAuNC0yLjEuNy0uMi40IDAgMS4zLjkgMi40TDYzLjEgOTBjLjkgMS4yIDIuMSAxLjggMy4zIDEuOHMyLjMtLjYgMy4xLTEuN2wxNC44LTE5LjNjLjktMS4xIDEuMS0yIC45LTIuNC0uMi0uMy0uOS0uNy0yLjEtLjdoLTcuNmMtLjkgMC0xLjctLjUtMi4xLTEuMi0uMy0uNC0uNC0uOC0uNC0xLjMgMC0xLjQgMS4xLTIuNSAyLjUtMi41aDcuNmMzLjEgMCA1LjUgMS4zIDYuNiAzLjVsLjMuN2MuNyAyLjEuMSA0LjYtMS43IDYuOXoiIGZpbGw9IiM0NDQiLz48L3N2Zz4=
  7. // @icon64 https://www.youxiaohou.com/64x64.png
  8. // @description 可以获取网盘文件真实下载地址。基于【网盘直链下载助手】修改自5.9.4版本,自用,去推广,修原有BUG,修改界面。
  9. // @license MIT
  10. // @homepage https://github.com/hmjz100/Online-disk-direct-link-download-assistant/
  11. // @supportURL https://github.com/hmjz100/Online-disk-direct-link-download-assistant/issues
  12. // @match *://pan.baidu.com/disk/home*
  13. // @match *://yun.baidu.com/disk/home*
  14. // @match *://pan.baidu.com/disk/main*
  15. // @match *://yun.baidu.com/disk/main*
  16. // @match *://pan.baidu.com/s/*
  17. // @match *://yun.baidu.com/s/*
  18. // @match *://pan.baidu.com/share/*
  19. // @match *://yun.baidu.com/share/*
  20. // @match *://www.aliyundrive.com/s/*
  21. // @match *://www.aliyundrive.com/drive*
  22. // @match *://cloud.189.cn/web/*
  23. // @match *://pan.xunlei.com/*
  24. // @match *://pan.quark.cn/*
  25. // @require https://unpkg.com/jquery@3.6.0/dist/jquery.min.js
  26. // @require https://cdn.jsdelivr.net/npm/sweetalert2@11/dist/sweetalert2.all.min.js
  27. // @require https://unpkg.com/js-md5@0.7.3/build/md5.min.js
  28. // @connect baidu.com
  29. // @connect baidupcs.com
  30. // @connect aliyundrive.com
  31. // @connect 189.cn
  32. // @connect xunlei.com
  33. // @connect quark.cn
  34. // @connect youxiaohou.com
  35. // @connect localhost
  36. // @connect *
  37. // @run-at document-idle
  38. // @grant unsafeWindow
  39. // @grant GM_xmlhttpRequest
  40. // @grant GM_setClipboard
  41. // @grant GM_setValue
  42. // @grant GM_getValue
  43. // @grant GM_openInTab
  44. // @grant GM_info
  45. // @grant GM_registerMenuCommand
  46. // @grant GM_cookie
  47. // ==/UserScript==
  48.  
  49. (function () {
  50. 'use strict';
  51. let pt = '', selectList = [], params = {}, mode = '', width = 800, pan = {}, color = '',
  52. doc = $(document), progress = {}, request = {}, ins = {}, idm = {};
  53. const scriptInfo = GM_info.script;//用于油小猴检测的脚本介绍
  54. const version = scriptInfo.version;//用于油小猴检测的脚本版本
  55. const author = scriptInfo.author;//用于油小猴检测的脚本作者
  56. const name = scriptInfo.name;//用于油小猴检测的脚本名称
  57. const customClass = {
  58. popup: 'pl-popup',
  59. header: 'pl-header',
  60. title: 'pl-title',
  61. closeButton: 'pl-close',
  62. content: 'pl-content',
  63. input: 'pl-input',
  64. footer: 'pl-footer'
  65. };//准备好要用到的元素
  66.  
  67. //加上SweetAlert的主题css
  68. //以后可能会直接把css本地化直接加到网页中
  69. let script = document.createElement('link');
  70. script.setAttribute('rel', 'stylesheet');
  71. script.setAttribute('type', 'text/css');
  72. script.href = "https://cdn.jsdelivr.net/gh/hmjz100/Online-disk-direct-link-download-assistant@main/default.min.css";
  73. document.documentElement.appendChild(script);
  74.  
  75.  
  76. //准备好Shell类型(用于curl下载)
  77. const terminalType = {
  78. wc: "Windows CMD",
  79. wp: "Windows PowerShell",
  80. lt: "Linux 终端",
  81. ls: "Linux Shell",
  82. mt: "MacOS 终端",
  83. };
  84.  
  85. //准备好信息界面的*假隐私设置*(实际上油小猴就是通过服务器获取的“暗号”)
  86. const getuserinfo = {
  87. yes: "允许",
  88. };
  89.  
  90. //准备好右上角的Toast提示
  91. let toast = Swal.mixin({
  92. toast: true,
  93. position: 'top-end',
  94. showConfirmButton: false,
  95. timer: 3500,
  96. timerProgressBar: true,
  97. didOpen: (toast) => {
  98. toast.addEventListener('mouseenter', Swal.stopTimer);
  99. toast.addEventListener('mouseleave', Swal.resumeTimer);
  100. }
  101. });
  102.  
  103. //提示的信息内容
  104. const message = {
  105. success: (text) => {
  106. toast.fire({title: text, icon: 'success'});
  107. },
  108. error: (text) => {
  109. toast.fire({title: text, icon: 'error'});
  110. },
  111. warning: (text) => {
  112. toast.fire({title: text, icon: 'warning'});
  113. },
  114. info: (text) => {
  115. toast.fire({title: text, icon: 'info'});
  116. },
  117. question: (text) => {
  118. toast.fire({title: text, icon: 'question'});
  119. }
  120. };
  121.  
  122. let base = {
  123. //脚本正常情况下默认加载的菜单
  124. registerMenuCommand() {
  125. GM_registerMenuCommand('⚙️ 设置', () => {
  126. this.showSetting();
  127. });
  128. GM_registerMenuCommand('更新日志', () => {
  129. this.showUpdateLog();
  130. });
  131. GM_registerMenuCommand('脚本信息', () => {
  132. this.showInfo();
  133. });
  134. GM_registerMenuCommand('取消点亮按钮', () => {
  135. this.registerSetting();
  136. });
  137. },
  138.  
  139. //取消点亮按钮按下后运行的
  140. registerSetting() {
  141. console.log("正在注入取消点亮按钮设置项目...");
  142. message.warning("正在注入取消点亮按钮设置项目...(您可以再次点亮按钮)");
  143. base.setValue('setting_init_code', 111111);
  144. history.go(0)
  145. },
  146.  
  147. //获取网页Cookie
  148. getCookie(name) {
  149. let arr = document.cookie.replace(/\s/g, "").split(';');
  150. for (let i = 0, l = arr.length; i < l; i++) {
  151. let tempArr = arr[i].split('=');
  152. if (tempArr[0] === name) {
  153. return decodeURIComponent(tempArr[1]);
  154. }
  155. }
  156. return '';
  157. },
  158.  
  159. isType(obj) {
  160. return Object.prototype.toString.call(obj).replace(/^\[object (.+)\]$/, '$1').toLowerCase();
  161. },
  162.  
  163. getValue(name) {
  164. return GM_getValue(name);
  165. },
  166.  
  167. setValue(name, value) {
  168. GM_setValue(name, value);
  169. },
  170.  
  171. getStorage(key) {
  172. try {
  173. return JSON.parse(localStorage.getItem(key));
  174. } catch (e) {
  175. return localStorage.getItem(key);
  176. }
  177. },
  178.  
  179. setStorage(key, value) {
  180. if (this.isType(value) === 'object' || this.isType(value) === 'array') {
  181. return localStorage.setItem(key, JSON.stringify(value));
  182. }
  183. return localStorage.setItem(key, value);
  184. },
  185.  
  186. setClipboard(text) {
  187. GM_setClipboard(text, 'text');
  188. },
  189.  
  190. e(str) {
  191. return btoa(unescape(encodeURIComponent(str)));
  192. },
  193.  
  194. d(str) {
  195. return decodeURIComponent(escape(atob(str)));
  196. },
  197.  
  198. getExtension(name) {
  199. const reg = /(?!\.)\w+$/;
  200. if (reg.test(name)) {
  201. let match = name.match(reg);
  202. return match[0].toUpperCase();
  203. }
  204. return '';
  205. },
  206.  
  207. sizeFormat(value) {
  208. if (value === +value) {
  209. let unit = ["B", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"];
  210. let index = Math.floor(Math.log(value) / Math.log(1024));
  211. let size = value / Math.pow(1024, index);
  212. size = size.toFixed(1);
  213. return size + unit[index];
  214. }
  215. return '';
  216. },
  217.  
  218. sortByName(arr) {
  219. const handle = () => {
  220. return (a, b) => {
  221. const p1 = a.filename ? a.filename : a.server_filename;
  222. const p2 = b.filename ? b.filename : b.server_filename;
  223. return p1.localeCompare(p2, "zh-CN");
  224. };
  225. };
  226. arr.sort(handle());
  227. },
  228.  
  229. fixFilename(name) {
  230. return name.replace(/[!?&|`"'*\/:<>\\]/g, '_');
  231. },
  232.  
  233. blobDownload(blob, filename) {
  234. if (blob instanceof Blob) {
  235. const url = URL.createObjectURL(blob);
  236. const a = document.createElement('a');
  237. a.href = url;
  238. a.download = filename;
  239. a.click();
  240. URL.revokeObjectURL(url);
  241. }
  242. },
  243.  
  244. post(url, data, headers, type) {
  245. if (this.isType(data) === 'object') {
  246. data = JSON.stringify(data);
  247. }
  248. return new Promise((resolve, reject) => {
  249. GM_xmlhttpRequest({
  250. method: "POST", url, headers, data,
  251. responseType: type || 'json',
  252. onload: (res) => {
  253. type === 'blob' ? resolve(res) : resolve(res.response || res.responseText);
  254. },
  255. onerror: (err) => {
  256. reject(err);
  257. },
  258. });
  259. });
  260. },
  261.  
  262. get(url, headers, type, extra) {
  263. return new Promise((resolve, reject) => {
  264. let requestObj = GM_xmlhttpRequest({
  265. method: "GET", url, headers,
  266. responseType: type || 'json',
  267. onload: (res) => {
  268. if (res.status === 204) {
  269. requestObj.abort();
  270. idm[extra.index] = true;
  271. }
  272. if (type === 'blob') {
  273. res.status === 200 && base.blobDownload(res.response, extra.filename);
  274. resolve(res);
  275. } else {
  276. resolve(res.response || res.responseText);
  277. }
  278. },
  279. onprogress: (res) => {
  280. if (extra && extra.filename && extra.index) {
  281. res.total > 0 ? progress[extra.index] = (res.loaded * 100 / res.total).toFixed(2) : progress[extra.index] = 0.00;
  282. }
  283. },
  284. onloadstart() {
  285. extra && extra.filename && extra.index && (request[extra.index] = requestObj);
  286. },
  287. onerror: (err) => {
  288. reject(err);
  289. },
  290. });
  291. });
  292. },
  293.  
  294. getFinalUrl(url, headers) {
  295. return new Promise((resolve, reject) => {
  296. let requestObj = GM_xmlhttpRequest({
  297. method: "GET", url, headers,
  298. onload: (res) => {
  299. resolve(res.finalUrl);
  300. },
  301. onerror: (err) => {
  302. reject(err);
  303. },
  304. });
  305. });
  306. },
  307.  
  308. addStyle(id, tag, css) {
  309. tag = tag || 'style';
  310. let doc = document, styleDom = doc.getElementById(id);
  311. if (styleDom) return;
  312. let style = doc.createElement(tag);
  313. style.rel = 'stylesheet';
  314. style.id = id;
  315. tag === 'style' ? style.innerHTML = css : style.href = css;
  316. doc.getElementsByTagName('head')[0].appendChild(style);
  317. },
  318.  
  319. sleep(time) {
  320. return new Promise(resolve => setTimeout(resolve, time));
  321. },
  322.  
  323. findReact(dom, traverseUp = 0) {
  324. const key = Object.keys(dom).find(key => {
  325. return key.startsWith("__reactFiber$")
  326. || key.startsWith("__reactInternalInstance$");
  327. });
  328. const domFiber = dom[key];
  329. if (domFiber == null) return null;
  330.  
  331. if (domFiber._currentElement) {
  332. let compFiber = domFiber._currentElement._owner;
  333. for (let i = 0; i < traverseUp; i++) {
  334. compFiber = compFiber._currentElement._owner;
  335. }
  336. return compFiber._instance;
  337. }
  338.  
  339. const GetCompFiber = fiber => {
  340. let parentFiber = fiber.return;
  341. while (typeof parentFiber.type == "string") {
  342. parentFiber = parentFiber.return;
  343. }
  344. return parentFiber;
  345. };
  346. let compFiber = GetCompFiber(domFiber);
  347. for (let i = 0; i < traverseUp; i++) {
  348. compFiber = GetCompFiber(compFiber);
  349. }
  350. return compFiber.stateNode || compFiber;
  351. },
  352.  
  353. //注册(不可用)默认设置
  354. initDefaultConfig() {
  355. let value = [{
  356. name: 'setting_rpc_domain',
  357. value: 'http://localhost'
  358. }, {
  359. name: 'setting_rpc_port',
  360. value: '16800'
  361. }, {
  362. name: 'setting_rpc_path',
  363. value: '/jsonrpc'
  364. }, {
  365. name: 'setting_rpc_token',
  366. value: ''
  367. }, {
  368. name: 'setting_rpc_dir',
  369. value: 'D:'
  370. }, {
  371. name: 'setting_terminal_type',
  372. value: 'wc'
  373. }, {
  374. name: 'setting_theme_color',
  375. value: '#574ab8'
  376. }, {
  377. name: 'setting_init_code',
  378. value: ''
  379. }, {
  380. name: 'setting_getuser_info',
  381. value: 'yes'
  382. }];
  383.  
  384. value.forEach((v) => {
  385. base.getValue(v.name) === undefined && base.setValue(v.name, v.value);
  386. });
  387. },
  388.  
  389. showSetting() {
  390. let dom = '', btn = '',
  391. colorList = ['#09AAFF', '#cc3235', '#526efa', '#518c17', '#ed944b', '#f969a5', '#bca280', '#574ab8', '#F8D800', '#0396FF', '#32CCBC', '#F6416C', '#2271b1', '#59524c', '#1d2327', '#18a497', '#10171d', '#2828ff'];
  392. dom += `<label class="pl-setting-label"><div class="pl-label">RPC主机</div><input type="text" placeholder="主机地址,需带上http(s)://" class="pl-input listener-domain" value="${base.getValue('setting_rpc_domain')}"></label>`;
  393. dom += `<label class="pl-setting-label"><div class="pl-label">RPC端口</div><input type="text" placeholder="端口号,例如:Motrix为16800" class="pl-input listener-port" value="${base.getValue('setting_rpc_port')}"></label>`;
  394. dom += `<label class="pl-setting-label"><div class="pl-label">RPC路径</div><input type="text" placeholder="路径,默认为/jsonrpc" class="pl-input listener-path" value="${base.getValue('setting_rpc_path')}"></label>`;
  395. dom += `<label class="pl-setting-label"><div class="pl-label">RPC密钥</div><input type="text" placeholder="无密钥无需填写" class="pl-input listener-token" value="${base.getValue('setting_rpc_token')}"></label>`;
  396. dom += `<label class="pl-setting-label"><div class="pl-label">保存路径</div><input type="text" placeholder="文件下载后保存路径,例如:D:" class="pl-input listener-dir" value="${base.getValue('setting_rpc_dir')}"></label>`;
  397.  
  398. colorList.forEach((v) => {
  399. btn += `<div data-color="${v}" style="background: ${v};border: 1px solid ${v}" class="pl-color-box listener-color ${v === base.getValue('setting_theme_color') ? 'checked' : ''}"></div>`;
  400. });
  401. dom += `<label class="pl-setting-label"><div class="pl-label">终端类型</div><select class="pl-input listener-terminal">`;
  402. Object.keys(terminalType).forEach(k => {
  403. dom += `<option value="${k}" ${base.getValue('setting_terminal_type') === k ? 'selected' : ''}>${terminalType[k]}</option>`;
  404. });
  405. dom += `</select></label>`;
  406. dom += `<label class="pl-setting-label"><div class="pl-label">主题颜色</div> <div class="pl-color">${btn}<div></label>`;
  407. dom = '<div>' + dom + '</div>';
  408.  
  409. Swal.fire({
  410. title: '助手设置',
  411. html: dom,
  412. icon: 'info',
  413. showCloseButton: true,
  414. showConfirmButton: false,
  415. footer: pan.footer,
  416. }).then(() => {
  417. message.success('设置成功!');
  418. history.go(0);
  419. });
  420.  
  421. doc.on('click', '.listener-color', async (e) => {
  422. base.setValue('setting_theme_color', e.target.dataset.color);
  423. message.success('设置成功!');
  424. history.go(0);
  425. });
  426. doc.on('input', '.listener-domain', async (e) => {
  427. base.setValue('setting_rpc_domain', e.target.value);
  428. });
  429. doc.on('input', '.listener-port', async (e) => {
  430. base.setValue('setting_rpc_port', e.target.value);
  431. });
  432. doc.on('input', '.listener-path', async (e) => {
  433. base.setValue('setting_rpc_path', e.target.value);
  434. });
  435. doc.on('input', '.listener-token', async (e) => {
  436. base.setValue('setting_rpc_token', e.target.value);
  437. });
  438. doc.on('input', '.listener-dir', async (e) => {
  439. base.setValue('setting_rpc_dir', e.target.value);
  440. });
  441. doc.on('change', '.listener-terminal', async (e) => {
  442. base.setValue('setting_terminal_type', e.target.value);
  443. });
  444. },
  445.  
  446. showInfo() {
  447. let hideinfo='';
  448. hideinfo +=`<label class="pl-setting-label"><div class="pl-label">版本</div>${version}</label>`;
  449. hideinfo +=`<label class="pl-setting-label"><div class="pl-label">作者</div>${author}</label>`;
  450. hideinfo += `<label class="pl-setting-label"><div class="pl-label">暗号</div>${pan.num}</label>`;
  451. hideinfo += `<label class="pl-setting-label"><div class="pl-label">代理</div>${pan.ua}</label>`;
  452. hideinfo += `<label class="pl-setting-label"><div class="pl-label">公众号</div>${pan.img}</label>`;
  453. hideinfo += `<label class="pl-setting-label"><div class="pl-label">网盘万能助手</div>${pan.assistant}</label>`;
  454. hideinfo += `<label class="pl-setting-label"><div class="pl-label">网盘镜像</div>${pan.mirror}</label>`;
  455. hideinfo += `<label class="pl-setting-label"><div class="pl-label">RPC在线管理</div>${pan.d}</label>`;
  456. hideinfo += `<label class="pl-setting-label"><div class="pl-label">IDM</div>${pan.idm}</label>`;
  457. hideinfo += `<label class="pl-setting-label"><div class="pl-label">提示文本</div>0${pan.init[0]}<br>1${pan.init[1]}<br>2${pan.init[2]}<br>3${pan.init[3]}<br>4${pan.init[4]}<br>5${pan.init[5]}</label>`;
  458. hideinfo += `<label class="pl-setting-label"><div class="pl-label">页脚</div>${pan.fotter}</label>`;
  459. hideinfo +=`<label class="pl-setting-label"><div class="pl-label">允许油小猴在线收集作者名称与获取暗号</div><select class="pl-input getuser_info">`;
  460. Object.keys(getuserinfo).forEach(value1 => {hideinfo += `<option value="${value1}" ${base.getValue('setting_getuser_info') === value1 ? 'selected' : ''}>${getuserinfo[value1]}</option></label>`;});
  461. hideinfo = '<div>' + hideinfo + '</div>';
  462.  
  463. Swal.fire({
  464. icon: 'info',
  465. title: '脚本信息',
  466. html: hideinfo,
  467. allowOutsideClick: false,
  468. showCloseButton: true,
  469. confirmButtonText: '保存配置'
  470. });
  471.  
  472. doc.on('change', '.getuser_info', async (e) => {
  473. base.setValue('setting_getuser_info', e.target.value);
  474. });
  475. },
  476.  
  477. showUpdateLog() {
  478. Swal.fire({
  479. icon: 'info',
  480. title: '更新日志(关闭按钮在下面哦)',
  481. html: '<span>V1.0.5.4<br>1、小修小改css,让主题色出现在更多地方;<br>2、修改下载链接获取失败的提示;<br>3、增加更多的主题色,可在助手设置查看;<br>4、homo彩蛋被删去力(悲)。<br>说明:此脚本不提供会员链接解析功能!<br><br>V1.0.5.3<br>1、修啦修啦,阿里云盘可以摸到下载菜单了。<br><br>V1.0.5.2<br>1、增加脚本信息菜单(没有用);<br>2、优化阿里云盘显示svg图片;<br>3、修改弹窗按钮颜色。<br><br>V1.0.5.1<br>1、修复在切换按钮主题后夸克网盘不能正常显示按钮。<br><br>V1.0.5<br>1、跟进官方V5.0.4版本;<br>2、小改动,照着官方版本更正文件名称检测;<br>3、保留彩蛋,但必须舍弃官方暗号。<br><br>V1.0.4<br>大改!<br>1、修复了原作者留下的夸克网盘切换文件夹就多一个“下载助手”按钮的大BUG;<br>2、终于来了,在下载菜单增加“助手设置”“更新日志”按钮;<br>【再也不用点进油猴管理再进设置了(保留油猴管理内设置)】<br>3、修改阿里云盘和夸克网盘下载助手按钮样式;<br>4、增加“取消点亮按钮”油猴菜单;<br>5、修改部分css,使其与选择的主题更贴切。<br><br>V1.0.3<br>1、增加一个小彩蛋; 提示:homo(需在未点亮按钮状态触发)<br>【需要重新恢复按钮为未点亮状态请进入 已安装脚本->编辑->开发者->重置到出厂->确定】<br>2、修改/增加默认主题色。<br><br>V1.0.2<br>1、修改并加宽界面,调整部分css,使Sweetalert2界面更美观,更与原版相近;<br>2、修改部分提示文字,使文字更容易复制。 <br><br>V1.0.1<br>1、去除更新提示;<br>2、更新Sweetalert2至11版本;<br>3、部分CDN节点更换为jsdelivr。<br><br>V1.0.0<br>1、增加“注入”功能(bushi);<br>2、去除广告。</span>',
  482. allowOutsideClick: false,
  483. showCloseButton: false,
  484. confirmButtonText: '我已阅',
  485. });
  486. },
  487.  
  488. createTip() {
  489. $('body').append('<div class="pl-tooltip"></div>');
  490.  
  491. doc.on('mouseenter mouseleave', '.listener-tip', (e) => {
  492. if (e.type === 'mouseenter') {
  493. let filename = e.currentTarget.innerText;
  494. let size = e.currentTarget.dataset.size;
  495. let tip = `${filename}<span style="margin-left: 10px;color: ${color}">${size}</span>`;
  496. $(e.currentTarget).css({opacity: '0.5'});
  497. $('.pl-tooltip').html(tip).css({
  498. 'left': e.pageX + 10 + 'px',
  499. 'top': e.pageY - e.currentTarget.offsetTop > 14 ? e.pageY + 'px' : e.pageY + 20 + 'px'
  500. }).show();
  501. } else {
  502. $(e.currentTarget).css({opacity: '1'});
  503. $('.pl-tooltip').hide(0);
  504. }
  505. });
  506. },
  507.  
  508. createLoading() {
  509. return $('<div class="pl-loading"><div class="pl-loading-box"><div><div></div><div></div></div></div></div>');
  510. },
  511.  
  512. createDownloadIframe() {
  513. let $div = $('<div style="padding:0;margin:0;display:block"></div>');
  514. let $iframe = $('<iframe src="javascript:;" id="downloadIframe" style="display:none"></iframe>');
  515. $div.append($iframe);
  516. $('body').append($div);
  517. },
  518.  
  519. getMirrorList(link, mirror, thread = 2) {
  520. let host = new URL(link).host;
  521. let mirrors = [];
  522. for (let i = 0; i < mirror.length; i++) {
  523. for (let j = 0; j < thread; j++) {
  524. let item = link.replace(host, mirror[i]) + '&'.repeat(j);
  525. mirrors.push(item);
  526. }
  527. }
  528. return mirrors.join('\n');
  529. },
  530.  
  531. addPanLinkerStyle() {
  532. color = base.getValue('setting_theme_color');
  533. let css = `
  534. body::-webkit-scrollbar { display: none }
  535. ::-webkit-scrollbar { width: 6px; height: 10px }
  536. ::-webkit-scrollbar-track { border-radius: 0; background: none }
  537. ::-webkit-scrollbar-thumb { background-color: rgba(85,85,85,.4) }
  538. ::-webkit-scrollbar-thumb,::-webkit-scrollbar-thumb:hover { border-radius: 5px; -webkit-box-shadow: inset 0 0 6px rgba(0,0,0,.2) }
  539. ::-webkit-scrollbar-thumb:hover { background-color: rgba(85,85,85,.3) }
  540. .swal2-popup { font-size: 16px !important; width: 550px;}
  541. .pl-popup { font-size: 12px !important; }
  542. .pl-popup a { color: ${color} !important; }
  543. .pl-header { padding: 0!important;align-items: flex-start!important; border-bottom: 1px solid #eee!important; margin: 0 0 10px!important; padding: 0 0 5px!important; }
  544. .pl-title { font-size: 16px!important; line-height: 1!important;white-space: nowrap!important; text-overflow: ellipsis!important;}
  545. .pl-content { padding: 0 !important; font-size: 12px!important; }
  546. .pl-main { max-height: 400px;overflow-y:scroll; }
  547. .pl-footer {font-size: 15px!important;justify-content: flex-start!important; margin: 10px 0 0!important; padding: 5px 0 0!important; color: #f56c6c!important; height:25px;}
  548. .pl-item { display: flex; align-items: center; line-height: 22px; }
  549. .pl-item-name { flex: 0 0 200px; text-align: left;margin-right: 10px; overflow: hidden; white-space: nowrap; text-overflow: ellipsis; cursor:default; height: 30px;}
  550. .pl-item-link { flex: 1; text-align: left; white-space: nowrap; text-overflow: ellipsis;cursor:pointer }
  551. .pl-item-btn { background: ${color}; padding: 4px 5px; border-radius: 3px; line-height: 1; cursor: pointer; color: #fff; }
  552. .pl-item-tip { display: flex; justify-content: space-between;flex: 1; }
  553. .pl-back { width: 70px; background: #ddd; border-radius: 3px; cursor:pointer; margin:1px 0; }
  554. .pl-ext { display: inline-block; width: 44px; background: #999; color: #fff; height: 16px; line-height: 16px; font-size: 12px; border-radius: 3px;}
  555. .pl-retry {padding: 3px 10px; background: #cc3235; color: #fff; border-radius: 3px; cursor: pointer;}
  556. .pl-browserdownload { padding: 3px 10px; background: ${color}; color: #fff; border-radius: 3px; cursor: pointer;}
  557. .pl-item-progress { display:flex;flex: 1;align-items:center}
  558. .pl-progress { display: inline-block;vertical-align: middle;width: 100%; box-sizing: border-box;line-height: 1;position: relative;height:15px; flex: 1}
  559. .pl-progress-outer { height: 15px;border-radius: 100px;background-color: #ebeef5;overflow: hidden;position: relative;vertical-align: middle;}
  560. .pl-progress-inner{ position: absolute;left: 0;top: 0;background-color: #409eff;text-align: right;border-radius: 100px;line-height: 1;white-space: nowrap;transition: width .6s ease;}
  561. .pl-progress-inner-text { display: inline-block;vertical-align: middle;color: #d1d1d1;font-size: 12px;margin: 0 5px;height: 15px}
  562. .pl-progress-tip{ flex:1;text-align:right}
  563. .pl-progress-how{ flex: 0 0 100px; background: #ddd; border-radius: 3px; margin-left: 10px; cursor: pointer; text-align: center;}
  564. .pl-progress-stop{ flex: 0 0 75px; padding: 0 10px; background: #cc3235; color: #fff; border-radius: 3px; cursor: pointer;margin-left:10px;height:20px}
  565. .pl-progress-inner-text:after { display: inline-block;content: "";height: 100%;vertical-align: middle;}
  566. .pl-btn-primary { background: ${color}; border: 0; border-radius: 4px; color: #ffffff; cursor: pointer; font-size: 12px; outline: none; display:flex; align-items: center; justify-content: center; margin: 2px 0; padding: 6px 0;transition: 0.3s opacity; }
  567. .pl-btn-primary:hover { opacity: 0.9;transition: 0.3s opacity; }
  568. .pl-btn-success { background: #55af28; animation: easeOpacity 1.2s 2; animation-fill-mode:forwards }
  569. .pl-btn-info { background: #606266; }
  570. .pl-btn-warning { background: #da9328; }
  571. .pl-btn-warning { background: #da9328; }
  572. .pl-btn-danger { background: #cc3235; }
  573. .ali-button {display: inline-flex;align-items: center;justify-content: center;border: 0 solid transparent;font-size: 14px;margin-left: 20px;padding: 1px 12px;position: relative;}
  574. .ali-button-big {display: inline-flex;align-items: center;justify-content: center;border: 0 solid transparent;border-radius: 5px;box-shadow: 0 0 0 0 transparent;width: fit-content;white-space: nowrap;flex-shrink: 0;font-size: 14px;line-height: 1.5;outline: 0;touch-action: manipulation;transition: background .3s ease,color .3s ease,border .3s ease,box-shadow .3s ease;color: #fff;background: ${color};margin-left: 20px;padding: 1px 12px;position: relative; cursor:pointer; height: 32px;}
  575. .ali-button:hover {background: rgb(122, 144, 255)}
  576. .tianyi-button {margin-right: 20px; padding: 4px 12px; border-radius: 4px; color: #fff; font-size: 12px; border: 1px solid ${color}; background: ${color}; cursor: pointer; position: relative;}
  577. .tianyi-button:hover {border-color: #a5a5a5; background: #a5a5a5;}
  578. .xunlei-button {display: inline-flex;align-items: center;justify-content: center;border: 0 solid transparent;border-radius: 5px;box-shadow: 0 0 0 0 transparent;width: fit-content;white-space: nowrap;flex-shrink: 0;font-size: 14px;line-height: 1.5;outline: 0;touch-action: manipulation;transition: background .3s ease,color .3s ease,border .3s ease,box-shadow .3s ease;color: #fff;background: ${color};margin-left: 12px;padding: 0px 12px;position: relative; cursor:pointer; height: 36px;}
  579. .xunlei-button:hover {background: #a5a5a5}
  580. .quark-button {display: inline-flex; align-items: center; justify-content: center; border: 1px solid #ddd; border-radius: 8px; white-space: nowrap; flex-shrink: 0; font-size: 14px; line-height: 1.5; outline: 0; color: #fff; margin-right: 10px; padding: 0px 14px; position: relative; cursor: pointer; height: 36px;}
  581. .quark-button:hover { background: #a5a5a5;}
  582. .pl-dropdown-menu {position: absolute;right: 0;top: 25px;padding: 5px 0;color: rgb(37, 38, 43);background: #fff;z-index: 999;width: 102px;border-radius: 10px;box-shadow: 0 0 1px 1px rgb(28 28 32 / 5%), 0 8px 24px rgb(28 28 32 / 12%);}
  583. .pl-dropdown-menu-old {position: absolute;right: 0;top: 30px;padding: 5px 0;color: rgb(37, 38, 43);background: #fff;z-index: 999;width: 102px;border: 1px solid #ddd;border-radius: 10px; box-shadow: 0 0 1px 1px rgb(28 28 32 / 5%), 0 8px 24px rgb(28 28 32 / 12%);}
  584. .pl-dropdown-menu-item { height: 30px;display: flex;align-items: center;justify-content: center;color: ${color};}
  585. .pl-dropdown-menu-item:hover { background-color: rgba(132,133,141,0.08);}
  586. .pl-button .pl-dropdown-menu { display: none; }
  587. .pl-button:hover .pl-dropdown-menu { display: block!important; }
  588. .pl-button-init { opacity: 0.5; animation: easeInitOpacity 1.2s 3; animation-fill-mode:forwards }
  589. @keyframes easeInitOpacity { from { opacity: 0.5; } 50% { opacity: 1 } to { opacity: 0.5; } }
  590. @keyframes easeOpacity { from { opacity: 1; } 50% { opacity: 0.35 } to { opacity: 1; } }
  591. .element-clicked { opacity: 0.5; }
  592. .pl-extra { margin-top: 10px;display:flex}
  593. .pl-extra button { flex: 1}
  594. .pointer { cursor:pointer }
  595. .pl-setting-label { display: flex;align-items: center;justify-content: space-between;padding-top: 10px; }
  596. .pl-label { flex: 0 0 100px;text-align:left; }
  597. .pl-input { flex: 1; padding: 8px 10px; border: 1px solid #c2c2c2; border-radius: 5px; font-size: 14px }
  598. .pl-color { flex: 1;display: flex;flex-wrap: wrap}
  599. .pl-color-box { width: 35px;height: 35px;margin:10px 10px 0 0;; box-sizing: border-box;border:1px solid #fff;cursor:pointer }
  600. .pl-color-box.checked { border:3px dashed #111!important }
  601. .pl-close:focus { outline: 0; box-shadow: none; }
  602. .tag-danger {color:#cc3235;margin: 0 5px;}
  603. .pl-tooltip { position: absolute; color: #ffffff; max-width: 600px; font-size: 12px; padding: 5px 10px; background: #333; border-radius: 5px; z-index: 110000; line-height: 1.3; display:none; word-break: break-all;}
  604. @keyframes load { 0% { transform: rotate(0deg) } 100% { transform: rotate(360deg) } }
  605. .pl-loading-box > div > div { position: absolute;border-radius: 50%;}
  606. .pl-loading-box > div > div:nth-child(1) { top: 9px;left: 9px;width: 82px;height: 82px;background: #ffffff;}
  607. .pl-loading-box > div > div:nth-child(2) { top: 14px;left: 38px;width: 25px;height: 25px;background: #666666;animation: load 1s linear infinite;transform-origin: 12px 36px;}
  608. .pl-loading { width: 16px;height: 16px;display: inline-block;overflow: hidden;background: none;}
  609. .pl-loading-box { width: 100%;height: 100%;position: relative;transform: translateZ(0) scale(0.16);backface-visibility: hidden;transform-origin: 0 0;}
  610. .pl-loading-box div { box-sizing: content-box; }
  611. .swal2-container { z-index:100000!important; }
  612. body.swal2-height-auto { height: inherit!important; }
  613. `;
  614. this.addStyle('panlinker-style', 'style', css);
  615. },
  616.  
  617. async initDialog() {
  618. let result = await Swal.fire({
  619. title: pan.init[0],
  620. allowOutsideClick: false,
  621. showCloseButton: true,
  622. showCancelButton: true,
  623. confirmButtonText: '确定',
  624. html: `<div><img style="width: 250px;margin-bottom: 10px;" src="${pan.img}" alt="${pan.img}"><input class="swal2-input" id="init" style="width:373;font-size:19px;" type="text" placeholder="${pan.init[1]}"><br><span>你可以选择“注入”立即点亮按钮,或者输入暗号。<br>(还是一样,判断没写好,直接注入吧)<br>暗号:“${pan.num}”。<br>(希望能有个人帮我看看判断问题出在哪儿,这句话在脚本的624行)</span></div>`,
  625. cancelButtonText: '注入(点亮下载助手按钮)'
  626. });
  627. if (result.isDismissed && result.dismiss === 'close') {console.log("窗口关闭");return};
  628. if (result.isDismissed && result.dismiss === 'cancel') {
  629. console.log("注入暗号")
  630. console.log("正在注入点亮按钮设置项目...");
  631. message.warning("正在注入点亮按钮设置项目...");
  632. setTimeout(() => {
  633. base.setValue('setting_init_code', pan.num);
  634. message.success("注入成功!");
  635. setTimeout(() => {
  636. message.success(pan.init[2]);
  637. setTimeout(() => {
  638. history.go(0);
  639. }, 3000);
  640. }, 3000);
  641. }, 5000);
  642. return;
  643. };
  644. if (pan.num === $('#init').val()) {
  645. console.log("暗号正确")
  646. message.success(pan.init[2]);
  647. setTimeout(() => {
  648. base.setValue('setting_init_code', pan.num);
  649. setTimeout(() => {
  650. history.go(0);
  651. }, 3000);
  652. }, 3000)
  653. return;
  654. } else {
  655. console.log("暗号错误")
  656. await Swal.fire({
  657. imageUrl: pan.img,
  658. title: pan.init[3],
  659. html: `<div><span>${pan.init[4]}<br>你可以在返回后选择“注入”立即点亮按钮,<br>或者在输入框键入以下暗号:“${pan.num}”。</span></div>`,
  660. confirmButtonText: '重新输入(返回)',
  661. });
  662. await this.initDialog();
  663. return;
  664. };
  665. /*---
  666. homo彩蛋被删去力(悲),存下图片罢!
  667. imageUrl: 'https://pic4.zhimg.com/80/v2-1b97a088e156c015108dec663bba8b04.jpg',
  668. imageUrl: 'https://lh1.hetaousercontent.com/img/7d4c1c0b4adb0e95.jpg',
  669. */
  670. },
  671. };
  672.  
  673. let baidu = {
  674.  
  675. _getExtra() {
  676. let seKey = decodeURIComponent(base.getCookie('BDCLND'));
  677. return '{' + '"sekey":"' + seKey + '"' + "}";
  678. },
  679.  
  680. _getSurl() {
  681. let reg = /(?<=s\/|surl=)([a-zA-Z0-9_-]+)/g;
  682. if (reg.test(location.href)) {
  683. return location.href.match(reg)[0];
  684. }
  685. return '';
  686. },
  687.  
  688. _getFidList() {
  689. let fidlist = [];
  690. selectList.forEach(v => {
  691. if (+v.isdir === 1) return;
  692. fidlist.push(v.fs_id);
  693. });
  694. return '[' + fidlist + ']';
  695. },
  696.  
  697. _resetData() {
  698. progress = {};
  699. $.each(request, (key) => {
  700. (request[key]).abort();
  701. });
  702. $.each(ins, (key) => {
  703. clearInterval(ins[key]);
  704. });
  705. idm = {};
  706. ins = {};
  707. request = {};
  708. },
  709.  
  710. setBDUSS() {
  711. try {
  712. GM_cookie && GM_cookie('list', {name: 'BDUSS'}, (cookies, error) => {
  713. if (!error) {
  714. base.setStorage("baiduyunPlugin_BDUSS", {BDUSS: cookies[0].value});
  715. }
  716. });
  717. } catch (e) {
  718. }
  719. },
  720.  
  721. getBDUSS() {
  722. let baiduyunPlugin_BDUSS = base.getStorage('baiduyunPlugin_BDUSS') ? base.getStorage('baiduyunPlugin_BDUSS') : '{"baiduyunPlugin_BDUSS":""}';
  723. return baiduyunPlugin_BDUSS.BDUSS || '';
  724. },
  725.  
  726. convertLinkToAria(link, filename, ua) {
  727. let BDUSS = this.getBDUSS();
  728. if (!!BDUSS) {
  729. filename = base.fixFilename(filename);
  730. return encodeURIComponent(`aria2c "${link}" --out "${filename}" --header "User-Agent: ${ua}" --header "Cookie: BDUSS=${BDUSS}"`);
  731. }
  732. return {
  733. link: pan.assistant,
  734. text: pan.init[5]
  735. };
  736. },
  737.  
  738. convertLinkToBC(link, filename, ua) {
  739. let BDUSS = this.getBDUSS();
  740. if (!!BDUSS) {
  741. let cookie = `BDUSS=${BDUSS}`;
  742. let bc = `AA/${encodeURIComponent(filename)}/?url=${encodeURIComponent(link)}&cookie=${encodeURIComponent(cookie)}&user_agent=${encodeURIComponent(ua)}ZZ`;
  743. return encodeURIComponent(`bc://http/${base.e(bc)}`);
  744. }
  745. return {
  746. link: pan.assistant,
  747. text: pan.init[5]
  748. };
  749. },
  750.  
  751. convertLinkToCurl(link, filename, ua) {
  752. let BDUSS = this.getBDUSS();
  753. if (!!BDUSS) {
  754. let terminal = base.getValue('setting_terminal_type');
  755. filename = base.fixFilename(filename);
  756. return encodeURIComponent(`${terminal !== 'wp' ? 'curl' : 'curl.exe'} -L -C - "${link}" -o "${filename}" -A "${ua}" -b "BDUSS=${BDUSS}"`);
  757. }
  758. return {
  759. link: pan.assistant,
  760. text: pan.init[5]
  761. };
  762. },
  763.  
  764. addPageListener() {
  765. function _factory(e) {
  766. let target = $(e.target);
  767. let item = target.parents('.pl-item');
  768. let link = item.find('.pl-item-link');
  769. let progress = item.find('.pl-item-progress');
  770. let tip = item.find('.pl-item-tip');
  771. return {
  772. item, link, progress, tip, target,
  773. };
  774. }
  775.  
  776. function _reset(i) {
  777. ins[i] && clearInterval(ins[i]);
  778. request[i] && request[i].abort();
  779. progress[i] = 0;
  780. idm[i] = false;
  781. }
  782.  
  783. doc.on('mouseenter mouseleave click', '.pl-button.g-dropdown-button', (e) => {
  784. if (e.type === 'mouseleave') {
  785. $(e.currentTarget).removeClass('button-open');
  786. } else {
  787. $(e.currentTarget).addClass('button-open');
  788. $(e.currentTarget).find('.pl-dropdown-menu').show();
  789. }
  790. });
  791. doc.on('mouseleave', '.pl-button.g-dropdown-button .pl-dropdown-menu', (e) => {
  792. $(e.currentTarget).hide();
  793. });
  794.  
  795. doc.on('click', '.pl-button-mode', (e) => {
  796. mode = e.target.dataset.mode;
  797. Swal.showLoading();
  798. this.getPCSLink();
  799. });
  800. doc.on('click', '.listener-link-api', async (e) => {
  801. e.preventDefault();
  802. let o = _factory(e);
  803. let $width = o.item.find('.pl-progress-inner');
  804. let $text = o.item.find('.pl-progress-inner-text');
  805. let filename = o.link[0].dataset.filename;
  806. let index = o.link[0].dataset.index;
  807. _reset(index);
  808. base.get(o.link[0].dataset.link, {"User-Agent": pan.ua}, 'blob', {filename, index});
  809. ins[index] = setInterval(() => {
  810. let prog = +progress[index] || 0;
  811. let isIDM = idm[index] || false;
  812. if (isIDM) {
  813. o.tip.hide();
  814. o.progress.hide();
  815. o.link.text('已成功唤起IDM,请查看IDM下载框!').animate({opacity: '0.5'}, "slow").show();
  816. clearInterval(ins[index]);
  817. idm[index] = false;
  818. } else {
  819. o.link.hide();
  820. o.tip.hide();
  821. o.progress.show();
  822. $width.css('width', prog + '%');
  823. $text.text(prog + '%');
  824. if (prog === 100) {
  825. clearInterval(ins[index]);
  826. progress[index] = 0;
  827. o.item.find('.pl-progress-stop').hide();
  828. o.item.find('.pl-progress-tip').html('下载完成,正在弹出浏览器下载框!');
  829. }
  830. }
  831. }, 500);
  832. });
  833. doc.on('click', '.listener-retry', async (e) => {
  834. let o = _factory(e);
  835. o.tip.hide();
  836. o.link.show();
  837. });
  838. doc.on('click', '.listener-how', async (e) => {
  839. let o = _factory(e);
  840. let index = o.link[0].dataset.index;
  841. if (request[index]) {
  842. request[index].abort();
  843. clearInterval(ins[index]);
  844. o.progress.hide();
  845. o.tip.show();
  846. }
  847.  
  848. });
  849. doc.on('click', '.listener-stop', async (e) => {
  850. let o = _factory(e);
  851. let index = o.link[0].dataset.index;
  852. if (request[index]) {
  853. request[index].abort();
  854. clearInterval(ins[index]);
  855. o.tip.hide();
  856. o.progress.hide();
  857. o.link.show(0);
  858. }
  859. });
  860. doc.on('click', '.listener-back', async (e) => {
  861. let o = _factory(e);
  862. o.tip.hide();
  863. o.link.show();
  864. });
  865. doc.on('click', '.listener-link-aria, .listener-copy-all', (e) => {
  866. e.preventDefault();
  867. if (!e.target.dataset.link) {
  868. $(e.target).removeClass('listener-copy-all').addClass('pl-btn-danger').html(`${pan.init[5]}👉<a href="${pan.assistant}" target="_blank" class="pl-a">点击此处安装</a>👈`);
  869. } else {
  870. base.setClipboard(decodeURIComponent(e.target.dataset.link));
  871. $(e.target).text('复制成功,快去粘贴吧!').animate({opacity: '0.5'}, "slow");
  872. }
  873. });
  874. doc.on('click', '.listener-link-rpc', async (e) => {
  875. let target = $(e.currentTarget);
  876. target.find('.icon').remove();
  877. target.find('.pl-loading').remove();
  878. target.prepend(base.createLoading());
  879. let res = await this.sendLinkToRPC(e.currentTarget.dataset.filename, e.currentTarget.dataset.link);
  880. if (res === 'success') {
  881. $('.listener-rpc-task').show();
  882. target.removeClass('pl-btn-danger').html('发送成功,快去看看吧!').animate({opacity: '0.5'}, "slow");
  883. } else if (res === 'assistant') {
  884. target.addClass('pl-btn-danger').html(`${pan.init[5]}👉<a href="${pan.assistant}" target="_blank" class="pl-a">点击此处安装</a>👈`);
  885. } else {
  886. target.addClass('pl-btn-danger').text('发送失败,请检查您的RPC配置信息!').animate({opacity: '0.5'}, "slow");
  887. }
  888. });
  889. doc.on('click', '.listener-send-rpc', (e) => {
  890. $('.listener-link-rpc').click();
  891. $(e.target).text('发送完成,发送结果见上方按钮!').animate({opacity: '0.5'}, "slow");
  892. });
  893. doc.on('click', '.listener-open-setting', () => {
  894. base.showSetting();
  895. });
  896. doc.on('click', '.listener-open-updatelog', () => {
  897. base.showUpdateLog();
  898. });
  899. doc.on('click', '.listener-rpc-task', () => {
  900. let rpc = JSON.stringify({
  901. domain: base.getValue('setting_rpc_domain'),
  902. port: base.getValue('setting_rpc_port'),
  903. }), url = `${pan.d}/?rpc=${base.e(rpc)}#${base.getValue('setting_rpc_token')}`;
  904. GM_openInTab(url, {active: true});
  905. });
  906. document.documentElement.addEventListener('mouseup', (e) => {
  907. if (e.target.nodeName === 'A' && ~e.target.className.indexOf('pl-a')) {
  908. e.stopPropagation();
  909. }
  910. }, true);
  911. },
  912.  
  913. addButton() {
  914. if (!pt) return;
  915. let $toolWrap;
  916. let $button = $(`<div class="g-dropdown-button pointer pl-button"><div style="color:#fff;background: ${color};border-color:${color}" class="g-button g-button-blue"><span class="g-button-right"><em class="icon icon-download"></em><span class="text" style="width: 60px;">下载助手</span></span></div><div class="menu" style="width:auto;z-index:41;border-color:${color}"><div class="g-button-menu pl-button-mode" data-mode="api" style="color:${color};">API下载</div><div class="g-button-menu pl-button-mode" data-mode="aria" style="color:${color};">Aria下载</div><div class="g-button-menu pl-button-mode" data-mode="rpc" style="color:${color};">RPC下载</div><div class="g-button-menu pl-button-mode" data-mode="curl" style="color:${color};">cURL下载</div><div class="g-button-menu pl-button-mode" data-mode="bc" style="color:${color};">BC下载</div><div class="g-button-menu pl-button-mode listener-open-setting" style="color:${color};">助手设置</div><div class="g-button-menu pl-button-mode listener-open-updatelog" style="color:${color};">更新日志</div></div></div>`);
  917. if (pt === 'home') $toolWrap = $(pan.btn.home);
  918. if (pt === 'main') {
  919. $toolWrap = $(pan.btn.main);
  920. $button = $(`<div class="pl-button" style="position: relative; display: inline-block; margin-right: 8px;"><button class="u-button u-button--primary u-button--small is-round is-has-icon" style="background: ${color};border-color: ${color};font-size: 14px; padding: 8px 16px; border: none;"><i class="u-icon u-icon-download"></i><span>下载助手</span></button><ul style="color:${color}" class="dropdown-list nd-common-float-menu pl-dropdown-menu"><li class="sub cursor-p pl-button-mode" data-mode="api">API下载</li><li class="sub cursor-p pl-button-mode" data-mode="aria">Aria下载</li><li class="sub cursor-p pl-button-mode" data-mode="rpc">RPC下载</li><li class="sub cursor-p pl-button-mode" data-mode="curl">cURL下载</li><li class="sub cursor-p pl-button-mode" data-mode="bc">BC下载</li><li class="sub cursor-p pl-button-mode listener-open-setting"">助手设置</li><li class="sub cursor-p pl-button-mode listener-open-updatelog">更新日志</li></ul></div>`);
  921. }
  922. if (pt === 'share') $toolWrap = $(pan.btn.share);
  923. $toolWrap.prepend($button);
  924. this.setBDUSS();
  925. this.addPageListener();
  926. },
  927.  
  928. addInitButton() {
  929. if (!pt) return;
  930. let $toolWrap;
  931. let $button = $(`<div class="g-dropdown-button pointer pl-button-init" style="opacity:.5"><div style="color:#fff;background: ${color};border-color:${color}" class="g-button g-button-blue"><span class="g-button-right"><em class="icon icon-download"></em><span class="text" style="width: 60px;">下载助手(未点亮)</span></span></div></div>`);
  932. if (pt === 'home') $toolWrap = $(pan.btn.home);
  933. if (pt === 'main') {
  934. $toolWrap = $(pan.btn.main);
  935. $button = $(`<div class="pl-button-init" style="opacity:.5; display: inline-block; margin-right: 8px;"><button class="u-button u-button--primary u-button--small is-round is-has-icon" style="background: ${color};border-color: ${color};font-size: 14px; padding: 8px 16px; border: none;"><i class="u-icon u-icon-download"></i><span>下载助手(未点亮)</span></button></div>`);
  936. }
  937. if (pt === 'share') $toolWrap = $(pan.btn.share);
  938. $toolWrap.prepend($button);
  939. $button.click(() => base.initDialog());
  940. },
  941.  
  942. async getPCSLink() {
  943. selectList = this.getSelectedList();
  944. let fidList = this._getFidList(), url, res;
  945.  
  946. if (pt === 'home' || pt === 'main') {
  947. if (selectList.length === 0) {
  948. return message.error('提示:请先勾选要下载的文件!');
  949. }
  950. if (fidList.length === 2) {
  951. return message.error('提示:请打开文件夹后勾选文件!');
  952. }
  953. fidList = encodeURIComponent(fidList);
  954. url = `${pan.pcs[0]}&fsids=${fidList}`;
  955. res = await base.get(url, {"User-Agent": pan.ua});
  956. }
  957. if (pt === 'share') {
  958. this.getShareData();
  959. if (selectList.length === 0) {
  960. return message.error('提示:请先勾选要下载的文件!');
  961. }
  962. if (fidList.length === 2) {
  963. return message.error('提示:请打开文件夹后勾选文件!');
  964. }
  965. if (!params.sign) {
  966. let url = `${pan.pcs[2]}&surl=${params.surl}&logid=${params.logid}`;
  967. let r = await base.get(url);
  968. if (r.errno === 0) {
  969. params.sign = r.data.sign;
  970. params.timestamp = r.data.timestamp;
  971. } else {
  972. let dialog = await Swal.fire({
  973. toast: true,
  974. icon: 'info',
  975. title: `提示:请将文件<span class="tag-danger">[保存到网盘]</span>👉前往<span class="tag-danger">[我的网盘]</span>中下载!`,
  976. showConfirmButton: true,
  977. confirmButtonText: '点击保存',
  978. position: 'top',
  979. });
  980. if (dialog.isConfirmed) {
  981. $('.tools-share-save-hb')[0].click();
  982. }
  983. return;
  984. }
  985. }
  986. if (!params.bdstoken) {
  987. return message.error('提示:请先登录(不可用)网盘!');
  988. }
  989. let formData = new FormData();
  990. formData.append('encrypt', params.encrypt);
  991. formData.append('product', params.product);
  992. formData.append('uk', params.uk);
  993. formData.append('primaryid', params.primaryid);
  994. formData.append('fid_list', fidList);
  995. formData.append('logid', params.logid);
  996. params.shareType === 'secret' ? formData.append('extra', params.extra) : '';
  997. url = `${pan.pcs[1]}&sign=${params.sign}&timestamp=${params.timestamp}`;
  998. res = await base.post(url, formData, {"User-Agent": pan.ua});
  999. }
  1000. if (res.errno === 0) {
  1001. let html = this.generateDom(res.list);
  1002. this.showMainDialog(pan[mode][0], html, pan[mode][1]);
  1003. } else if (res.errno === 112) {
  1004. return message.error('提示:页面过期,请刷新重试!');
  1005. } else {
  1006. message.error('提示:获取下载链接失败!请刷新网页后重试!(或者试试重新登录(不可用)网盘?)');
  1007. }
  1008. },
  1009.  
  1010. generateDom(list) {
  1011. let content = '<div class="pl-main">';
  1012. let alinkAllText = '';
  1013. base.sortByName(list);
  1014. list.forEach((v, i) => {
  1015. if (v.isdir === 1) return;
  1016. let filename = v.server_filename || v.filename;
  1017. let ext = base.getExtension(filename);
  1018. let size = base.sizeFormat(v.size);
  1019. let dlink = v.dlink;
  1020. if (mode === 'api') {
  1021. content += `<div class="pl-item">
  1022. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1023. <a class="pl-item-link pl-a listener-link-api" href="${dlink}" data-filename="${filename}" data-link="${dlink}" data-index="${i}">${dlink}<br>下载 ${filename}</a>
  1024. <div class="pl-item-tip" style="display: none"><span>若没有弹出IDM下载框,找到IDM <b>选项</b> -> <b>文件类型</b> -> <b>第一个框</b> 中添加后缀 <span class="pl-ext">${ext}</span>,<a href="${pan.idm}" target="_blank" class="pl-a">详见此处</a></span> <span class="pl-back listener-back">返回</span></div>
  1025. <div class="pl-item-progress" style="display: none">
  1026. <div class="pl-progress">
  1027. <div class="pl-progress-outer"></div>
  1028. <div class="pl-progress-inner" style="width:5%">
  1029. <div class="pl-progress-inner-text">0%</div>
  1030. </div>
  1031. </div>
  1032. <span class="pl-progress-stop listener-stop">取消下载</span>
  1033. <span class="pl-progress-tip">未发现IDM,使用自带浏览器下载</span>
  1034. <span class="pl-progress-how listener-how">如何唤起IDM?</span>
  1035. </div></div>`;
  1036. }
  1037. if (mode === 'aria') {
  1038. let alink = this.convertLinkToAria(dlink, filename, pan.ua);
  1039. if (typeof (alink) === 'object') {
  1040. content += `<div class="pl-item">
  1041. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1042. <a class="pl-item-link pl-a" target="_blank" href="${alink.link}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  1043. } else {
  1044. alinkAllText += alink + '\r\n';
  1045. content += `<div class="pl-item">
  1046. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1047. <a class="pl-item-link pl-a listener-link-aria" href="${alink}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1048. }
  1049. }
  1050. if (mode === 'rpc') {
  1051. content += `<div class="pl-item">
  1052. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1053. <button class="pl-item-link listener-link-rpc pl-btn-primary pl-btn-info" data-filename="${filename}" data-link="${dlink}"><em class="icon icon-device"></em><span style="margin-left: 5px;">推送到 RPC 下载器</span></button></div>`;
  1054. }
  1055. if (mode === 'curl') {
  1056. let alink = this.convertLinkToCurl(dlink, filename, pan.ua);
  1057. if (typeof (alink) === 'object') {
  1058. content += `<div class="pl-item">
  1059. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1060. <a class="pl-item-link pl-a" target="_blank" href="${alink.link}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  1061. } else {
  1062. alinkAllText += alink + '\r\n';
  1063. content += `<div class="pl-item">
  1064. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1065. <a class="pl-item-link pl-a listener-link-aria" href="${alink}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1066. }
  1067. }
  1068. if (mode === 'bc') {
  1069. let alink = this.convertLinkToBC(dlink, filename, pan.ua);
  1070. if (typeof (alink) === 'object') {
  1071. content += `<div class="pl-item">
  1072. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1073. <a class="pl-item-link pl-a" href="${decodeURIComponent(alink.link)}" title="点击用比特彗星下载" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink.text)}</a> </div>`;
  1074. } else {
  1075. alinkAllText += alink + '\r\n';
  1076. content += `<div class="pl-item">
  1077. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1078. <a class="pl-item-link pl-a" href="${decodeURIComponent(alink)}" title="点击用比特彗星下载" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1079. }
  1080. }
  1081. });
  1082. content += '</div>';
  1083. if (mode === 'aria')
  1084. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button></div>`;
  1085. if (mode === 'rpc') {
  1086. let rpc = base.getValue('setting_rpc_domain') + ':' + base.getValue('setting_rpc_port') + base.getValue('setting_rpc_path');
  1087. content += `<div class="pl-extra"><button class="pl-btn-primary listener-send-rpc">发送全部链接</button><button title="${rpc}" class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px">设置 RPC 参数(当前为:${rpc})</button><button class="pl-btn-primary pl-btn-success listener-rpc-task" style="margin-left: 10px;display: none">查看下载任务</button></div>`;
  1088. }
  1089. if (mode === 'curl')
  1090. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button><button class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px;">设置终端类型(当前为:${terminalType[base.getValue('setting_terminal_type')]})</button></div>`;
  1091. return content;
  1092. },
  1093.  
  1094. async sendLinkToRPC(filename, link) {
  1095. let rpc = {
  1096. domain: base.getValue('setting_rpc_domain'),
  1097. port: base.getValue('setting_rpc_port'),
  1098. path: base.getValue('setting_rpc_path'),
  1099. token: base.getValue('setting_rpc_token'),
  1100. dir: base.getValue('setting_rpc_dir'),
  1101. };
  1102. let BDUSS = this.getBDUSS();
  1103. if (!BDUSS) return 'assistant';
  1104.  
  1105. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  1106. let rpcData = {
  1107. id: new Date().getTime(),
  1108. jsonrpc: '2.0',
  1109. method: 'aria2.addUri',
  1110. params: [`token:${rpc.token}`, [link], {
  1111. dir: rpc.dir,
  1112. out: filename,
  1113. header: [`User-Agent: ${pan.ua}`, `Cookie: BDUSS=${BDUSS}`]
  1114. }]
  1115. };
  1116. try {
  1117. let res = await base.post(url, rpcData, {"User-Agent": pan.ua}, '');
  1118. if (res.result) return 'success';
  1119. return 'fail';
  1120. } catch (e) {
  1121. return 'fail';
  1122. }
  1123. },
  1124.  
  1125. getSelectedList() {
  1126. try {
  1127. return require('system-core:context/context.js').instanceForSystem.list.getSelected();
  1128. } catch (e) {
  1129. return document.querySelector('.wp-s-core-pan').__vue__.selectedList;
  1130. }
  1131. },
  1132.  
  1133. getLogid() {
  1134. let ut = require("system-core:context/context.js").instanceForSystem.tools.baseService;
  1135. return ut.base64Encode(base.getCookie("BAIDUID"));
  1136. },
  1137.  
  1138. getShareData() {
  1139. let res = locals.dump();
  1140. params.shareType = 'secret';
  1141. params.sign = '';
  1142. params.timestamp = '';
  1143. params.bdstoken = res.bdstoken.value;
  1144. params.channel = 'chunlei';
  1145. params.clienttype = 0;
  1146. params.web = 1;
  1147. params.app_id = 250528;
  1148. params.encrypt = 0;
  1149. params.product = 'share';
  1150. params.logid = this.getLogid();
  1151. params.primaryid = res.shareid.value;
  1152. params.uk = res.share_uk.value;
  1153. params.shareType === 'secret' && (params.extra = this._getExtra());
  1154. params.surl = this._getSurl();
  1155. },
  1156.  
  1157. detectPage() {
  1158. let path = location.pathname;
  1159. if (/^\/disk\/home/.test(path)) return 'home';
  1160. if (/^\/disk\/main/.test(path)) return 'main';
  1161. if (/^\/(s|share)\//.test(path)) return 'share';
  1162. return '';
  1163. return '';
  1164. },
  1165.  
  1166. showMainDialog(title, html, footer) { //下载窗口
  1167. Swal.fire({
  1168. title,
  1169. html,
  1170. footer,
  1171. allowOutsideClick: false,
  1172. showCloseButton: true,
  1173. confirmButtonText: '关闭',
  1174. position: 'top',
  1175. width: '1000px',
  1176. padding: '15px 20px 5px',
  1177. customClass,
  1178. }).then(() => {
  1179. this._resetData();
  1180. });
  1181. },
  1182.  
  1183. async initPanLinker() {
  1184. base.initDefaultConfig();
  1185. base.addPanLinkerStyle();
  1186. pt = this.detectPage();
  1187. if (base.getValue('setting_getuser_info') === 'yes') {
  1188. let res = await base.post
  1189. (`https://api.youxiaohou.com/config?ver=${version}&a=${author}`, {}, {}, 'text');
  1190. pan = JSON.parse(base.d(res));
  1191. };
  1192. Object.freeze && Object.freeze(pan);
  1193. pan.num === base.getValue('setting_init_code') ? this.addButton() : this.addInitButton();
  1194. base.createTip();
  1195. base.registerMenuCommand();
  1196. }
  1197. };
  1198.  
  1199. let ali = {
  1200.  
  1201. convertLinkToAria(link, filename, ua) {
  1202. filename = base.fixFilename(filename);
  1203. return encodeURIComponent(`aria2c "${link}" --out "${filename}" --header "Referer: https://www.aliyundrive.com/"`);
  1204. },
  1205.  
  1206. convertLinkToBC(link, filename, ua) {
  1207. let bc = `AA/${encodeURIComponent(filename)}/?url=${encodeURIComponent(link)}&refer=${encodeURIComponent('https://www.aliyundrive.com/')}ZZ`;
  1208. return encodeURIComponent(`bc://http/${base.e(bc)}`);
  1209. },
  1210.  
  1211. convertLinkToCurl(link, filename, ua) {
  1212. let terminal = base.getValue('setting_terminal_type');
  1213. filename = base.fixFilename(filename);
  1214. return encodeURIComponent(`${terminal !== 'wp' ? 'curl' : 'curl.exe'} -L -C - "${link}" -o "${filename}" -e "https://www.aliyundrive.com/"`);
  1215. },
  1216.  
  1217. addPageListener() {
  1218. doc.on('click', '.pl-button-mode', (e) => {
  1219. mode = e.target.dataset.mode;
  1220. Swal.showLoading();
  1221. this.getPCSLink();
  1222. });
  1223. doc.on('click', '.listener-link-api', async (e) => {
  1224. e.preventDefault();
  1225. let dataset = e.currentTarget.dataset;
  1226. let href = dataset.link;
  1227. let url = await this.getRealLink(dataset.did, dataset.fid);
  1228. if (url) href = url;
  1229. let d = document.createElement("a");
  1230. d.download = e.currentTarget.dataset.filename;
  1231. d.rel = "noopener";
  1232. d.href = href;
  1233. d.dispatchEvent(new MouseEvent("click"));
  1234. });
  1235. doc.on('click', '.listener-link-aria, .listener-copy-all', (e) => {
  1236. e.preventDefault();
  1237. base.setClipboard(decodeURIComponent(e.target.dataset.link));
  1238. $(e.target).text('复制成功,快去粘贴吧!').animate({opacity: '0.5'}, "slow");
  1239. });
  1240. doc.on('click', '.listener-link-rpc', async (e) => {
  1241. let target = $(e.currentTarget);
  1242. target.find('.icon').remove();
  1243. target.find('.pl-loading').remove();
  1244. target.prepend(base.createLoading());
  1245. let res = await this.sendLinkToRPC(e.currentTarget.dataset.filename, e.currentTarget.dataset.link);
  1246. if (res === 'success') {
  1247. $('.listener-rpc-task').show();
  1248. target.removeClass('pl-btn-danger').html('发送成功,快去看看吧!').animate({opacity: '0.5'}, "slow");
  1249. } else {
  1250. target.addClass('pl-btn-danger').text('发送失败,请检查您的RPC配置信息!').animate({opacity: '0.5'}, "slow");
  1251. }
  1252. });
  1253. doc.on('click', '.listener-send-rpc', (e) => {
  1254. $('.listener-link-rpc').click();
  1255. $(e.target).text('发送完成,发送结果见上方按钮!').animate({opacity: '0.5'}, "slow");
  1256. });
  1257. doc.on('click', '.listener-open-setting', () => {
  1258. base.showSetting();
  1259. });
  1260. doc.on('click', '.listener-open-updatelog', () => {
  1261. base.showUpdateLog();
  1262. });
  1263. doc.on('click', '.listener-rpc-task', () => {
  1264. let rpc = JSON.stringify({
  1265. domain: base.getValue('setting_rpc_domain'),
  1266. port: base.getValue('setting_rpc_port'),
  1267. }), url = `${pan.d}/?rpc=${base.e(rpc)}#${base.getValue('setting_rpc_token')}`;
  1268. GM_openInTab(url, {active: true});
  1269. });
  1270. },
  1271.  
  1272. async getRealLink(d, f) {
  1273. let authorization = `${base.getStorage('token').token_type} ${base.getStorage('token').access_token}`;
  1274. let res = await base.post(pan.pcs[1], {
  1275. drive_id: d,
  1276. file_id: f
  1277. }, {
  1278. authorization,
  1279. "content-type": "application/json;charset=utf-8",
  1280. });
  1281. if (res.url) {
  1282. return res.url;
  1283. }
  1284. return '';
  1285. },
  1286.  
  1287. addButton() {
  1288. if (!pt) return;
  1289. let $toolWrap;
  1290. let $button = $(`<div class="ali-button-big">下载助手<div class="button--3S7z9 ali-button pl-button"><span data-role="icon" data-render-as="svg" class="icon"><svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M853.333 938.667H170.667a85.333 85.333 0 0 1-85.334-85.334v-384A85.333 85.333 0 0 1 170.667 384H288a32 32 0 0 1 0 64H170.667a21.333 21.333 0 0 0-21.334 21.333v384a21.333 21.333 0 0 0 21.334 21.334h682.666a21.333 21.333 0 0 0 21.334-21.334v-384A21.333 21.333 0 0 0 853.333 448H736a32 32 0 0 1 0-64h117.333a85.333 85.333 0 0 1 85.334 85.333v384a85.333 85.333 0 0 1-85.334 85.334z" fill="#FFFFFF"></path><path d="M715.03 543.552a32.81 32.81 0 0 0-46.251 0L554.005 657.813v-540.48a32 32 0 0 0-64 0v539.734L375.893 543.488a32.79 32.79 0 0 0-46.229 0 32.427 32.427 0 0 0 0 46.037l169.557 168.811a32.81 32.81 0 0 0 46.251 0l169.557-168.81a32.47 32.47 0 0 0 0-45.974z" fill="#FFFFFF"></path></svg></span><ul class="pl-dropdown-menu"><li class="pl-dropdown-menu-item pl-button-mode" data-mode="api">API下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="aria" >Aria下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="rpc">RPC下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="curl">cURL下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="bc" >BC下载</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-setting">助手设置</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-updatelog">更新日志</li></ul></div></div>`);
  1291. if (pt === 'home') {
  1292. let ins = setInterval(() => {
  1293. $toolWrap = $(pan.btn.home);
  1294. if ($toolWrap.length > 0) {
  1295. $toolWrap.append($button);
  1296. clearInterval(ins);
  1297. }
  1298. }, 50);
  1299. }
  1300. if (pt === 'share') {
  1301. $button.css({'margin-right': '10px'});
  1302. let ins = setInterval(() => {
  1303. $toolWrap = $(pan.btn.share);
  1304. if ($toolWrap.length > 0) {
  1305. $toolWrap.prepend($button);
  1306. clearInterval(ins);
  1307. }
  1308. }, 50);
  1309. }
  1310. base.createDownloadIframe();
  1311. this.addPageListener();
  1312. },
  1313.  
  1314. addInitButton() {
  1315. if (!pt) return;
  1316. let $toolWrap;
  1317. let $button = $(`<div class="ali-button-big">下载助手(未点亮)<div class="button--3S7z9 ali-button pl-button-init"><span data-role="icon" data-render-as="svg" class="icon"><svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg" width="16" height="16"><path d="M853.333 938.667H170.667a85.333 85.333 0 0 1-85.334-85.334v-384A85.333 85.333 0 0 1 170.667 384H288a32 32 0 0 1 0 64H170.667a21.333 21.333 0 0 0-21.334 21.333v384a21.333 21.333 0 0 0 21.334 21.334h682.666a21.333 21.333 0 0 0 21.334-21.334v-384A21.333 21.333 0 0 0 853.333 448H736a32 32 0 0 1 0-64h117.333a85.333 85.333 0 0 1 85.334 85.333v384a85.333 85.333 0 0 1-85.334 85.334z" fill="#FFFFFF"></path><path d="M715.03 543.552a32.81 32.81 0 0 0-46.251 0L554.005 657.813v-540.48a32 32 0 0 0-64 0v539.734L375.893 543.488a32.79 32.79 0 0 0-46.229 0 32.427 32.427 0 0 0 0 46.037l169.557 168.811a32.81 32.81 0 0 0 46.251 0l169.557-168.81a32.47 32.47 0 0 0 0-45.974z" fill="#FFFFFF"></path></svg></span></div>`);
  1318. if (pt === 'home') {
  1319. let ins = setInterval(() => {
  1320. $toolWrap = $(pan.btn.home);
  1321. if ($toolWrap.length > 0) {
  1322. $toolWrap.append($button);
  1323. clearInterval(ins);
  1324. }
  1325. }, 50);
  1326. }
  1327. if (pt === 'share') {
  1328. $button.css({'margin-right': '10px'});
  1329. let ins = setInterval(() => {
  1330. $toolWrap = $(pan.btn.share);
  1331. if ($toolWrap.length > 0) {
  1332. $toolWrap.prepend($button);
  1333. clearInterval(ins);
  1334. }
  1335. }, 50);
  1336. }
  1337. $button.click(() => base.initDialog());
  1338. },
  1339.  
  1340. async getPCSLink() {
  1341. let reactDomGrid = document.getElementsByClassName(pan.dom.grid)[0];
  1342. if (reactDomGrid) {
  1343. let res = await Swal.fire({
  1344. title: '提示',
  1345. html: '<div style="display: flex;align-items: center;justify-content: center;">请先切换到&nbsp;&nbsp;<b>列表视图</b>&nbsp;“<svg class="icon" viewBox="0 0 1024 1024" width="20" height="20"><use xlink:href="#PDSDrag"></use></svg>”&nbsp;&nbsp;后获取下载链接!</div>',
  1346. icon: 'info',
  1347. confirmButtonText: '点击切换'
  1348. });
  1349. if (res) {
  1350. $('.switch-wrapper--1yEfx').trigger('click');
  1351. return message.success('切换成功,请重新获取下载链接!');
  1352. }
  1353. return false;
  1354. }
  1355. selectList = this.getSelectedList();
  1356. if (selectList.length === 0) {
  1357. return message.error('提示:请先勾选要下载的文件!');
  1358. }
  1359. if (this.isOnlyFolder()) {
  1360. return message.error('提示:请打开文件夹后勾选文件!');
  1361. }
  1362. if (pt === 'share') {
  1363. if (selectList.length > 20) {
  1364. return message.error('提示:单次最多可勾选 20 个文件!');
  1365. }
  1366. try {
  1367. let authorization = `${base.getStorage('token').token_type} ${base.getStorage('token').access_token}`;
  1368. let xShareToken = base.getStorage('shareToken').share_token;
  1369.  
  1370. for (let i = 0; i < selectList.length; i++) {
  1371. let res = await base.post(pan.pcs[0], {
  1372. expire_sec: 600,
  1373. file_id: selectList[i].fileId,
  1374. share_id: selectList[i].shareId
  1375. }, {
  1376. authorization,
  1377. "content-type": "application/json;charset=utf-8",
  1378. "x-share-token": xShareToken
  1379. });
  1380. if (res.download_url) {
  1381. selectList[i].downloadUrl = res.download_url;
  1382. }
  1383. }
  1384. } catch (e) {
  1385. return message.error('提示:请先登录(不可用)网盘!');
  1386. }
  1387. }
  1388. let html = this.generateDom(selectList);
  1389. this.showMainDialog(pan[mode][0], html, pan[mode][1]);
  1390. },
  1391.  
  1392. generateDom(list) {
  1393. let content = '<div class="pl-main">';
  1394. let alinkAllText = '';
  1395. list.forEach((v, i) => {
  1396. if (v.type === 'folder') return;
  1397. let filename = v.name;
  1398. let fid = v.fileId;
  1399. let did = v.driveId;
  1400. let size = base.sizeFormat(v.size);
  1401. let dlink = v.downloadUrl || v.url;
  1402. if (mode === 'api') {
  1403. content += `<div class="pl-item">
  1404. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1405. <a class="pl-item-link listener-link-api" data-did="${did}" data-fid="${fid}" data-filename="${filename}" data-link="${dlink}" data-index="${i}">${dlink}</a>
  1406. </div>`;
  1407. }
  1408. if (mode === 'aria') {
  1409. let alink = this.convertLinkToAria(dlink, filename, navigator.userAgent);
  1410. if (typeof (alink) === 'object') {
  1411. content += `<div class="pl-item">
  1412. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1413. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  1414. } else {
  1415. alinkAllText += alink + '\r\n';
  1416. content += `<div class="pl-item">
  1417. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1418. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1419. }
  1420. }
  1421. if (mode === 'rpc') {
  1422. content += `<div class="pl-item">
  1423. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1424. <button class="pl-item-link listener-link-rpc pl-btn-primary pl-btn-info" data-filename="${filename}" data-link="${dlink}"><em class="icon icon-device"></em><span style="margin-left: 5px;">推送到 RPC 下载器</span></button></div>`;
  1425. }
  1426. if (mode === 'curl') {
  1427. let alink = this.convertLinkToCurl(dlink, filename, navigator.userAgent);
  1428. if (typeof (alink) === 'object') {
  1429. content += `<div class="pl-item">
  1430. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1431. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  1432. } else {
  1433. alinkAllText += alink + '\r\n';
  1434. content += `<div class="pl-item">
  1435. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1436. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1437. }
  1438. }
  1439. if (mode === 'bc') {
  1440. let alink = this.convertLinkToBC(dlink, filename, navigator.userAgent);
  1441. content += `<div class="pl-item">
  1442. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1443. <a class="pl-item-link" href="${decodeURIComponent(alink)}" title="点击用比特彗星下载" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1444. }
  1445. });
  1446. content += '</div>';
  1447. if (mode === 'aria')
  1448. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button></div>`;
  1449. if (mode === 'rpc') {
  1450. let rpc = base.getValue('setting_rpc_domain') + ':' + base.getValue('setting_rpc_port') + base.getValue('setting_rpc_path');
  1451. content += `<div class="pl-extra"><button class="pl-btn-primary listener-send-rpc">发送全部链接</button><button title="${rpc}" class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px">设置 RPC 参数(当前为:${rpc})</button><button class="pl-btn-primary pl-btn-success listener-rpc-task" style="margin-left: 10px;display: none">查看下载任务</button></div>`;
  1452. }
  1453. if (mode === 'curl')
  1454. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button><button class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px;">设置终端类型(当前为:${terminalType[base.getValue('setting_terminal_type')]})</button></div>`;
  1455. return content;
  1456. },
  1457.  
  1458. async sendLinkToRPC(filename, link) {
  1459. let rpc = {
  1460. domain: base.getValue('setting_rpc_domain'),
  1461. port: base.getValue('setting_rpc_port'),
  1462. path: base.getValue('setting_rpc_path'),
  1463. token: base.getValue('setting_rpc_token'),
  1464. dir: base.getValue('setting_rpc_dir'),
  1465. };
  1466.  
  1467. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  1468. let rpcData = {
  1469. id: new Date().getTime(),
  1470. jsonrpc: '2.0',
  1471. method: 'aria2.addUri',
  1472. params: [`token:${rpc.token}`, [link], {
  1473. dir: rpc.dir,
  1474. out: filename,
  1475. header: [`Referer: https://www.aliyundrive.com/`]
  1476. }]
  1477. };
  1478. try {
  1479. let res = await base.post(url, rpcData, {"Referer": "https://www.aliyundrive.com/"}, '');
  1480. if (res.result) return 'success';
  1481. return 'fail';
  1482. } catch (e) {
  1483. return 'fail';
  1484. }
  1485. },
  1486.  
  1487. getSelectedList() {
  1488. try {
  1489. let selectedList = [];
  1490. let reactDom = document.getElementsByClassName(pan.dom.list)[0];
  1491. let reactObj = base.findReact(reactDom, 1);
  1492. let props = reactObj.pendingProps;
  1493. if (props) {
  1494. let fileList = props.dataSource || [];
  1495. let selectedKeys = props.selectedKeys.split(',');
  1496. fileList.forEach((val) => {
  1497. if (selectedKeys.includes(val.fileId)) {
  1498. selectedList.push(val);
  1499. }
  1500. });
  1501. }
  1502. return selectedList;
  1503. } catch (e) {
  1504. return [];
  1505. }
  1506. },
  1507.  
  1508. detectPage() {
  1509. let path = location.pathname;
  1510. if (/^\/(drive)/.test(path)) return 'home';
  1511. if (/^\/(s|share)\//.test(path)) return 'share';
  1512. return '';
  1513. },
  1514.  
  1515. isOnlyFolder() {
  1516. for (let i = 0; i < selectList.length; i++) {
  1517. if (selectList[i].type === 'file') return false;
  1518. }
  1519. return true;
  1520. },
  1521.  
  1522. showMainDialog(title, html, footer) {
  1523. Swal.fire({
  1524. title,
  1525. html,
  1526. footer,
  1527. allowOutsideClick: false,
  1528. showCloseButton: true,
  1529. showConfirmButton: false,
  1530. position: 'top',
  1531. width,
  1532. padding: '15px 20px 5px',
  1533. customClass,
  1534. });
  1535. },
  1536.  
  1537. async initPanLinker() {
  1538. base.initDefaultConfig();
  1539. base.addPanLinkerStyle();
  1540. pt = this.detectPage();
  1541. if (base.getValue('setting_getuser_info') === 'yes') {
  1542. let res = await base.post
  1543. (`https://api.youxiaohou.com/config/ali?ver=${version}&a=${author}`, {}, {}, 'text');
  1544. pan = JSON.parse(base.d(res));
  1545. };
  1546. Object.freeze && Object.freeze(pan);
  1547. pan.num === base.getValue('setting_init_code') ? this.addButton() : this.addInitButton();
  1548. base.createTip();
  1549. base.registerMenuCommand();
  1550. }
  1551. };
  1552.  
  1553. let tianyi = {
  1554.  
  1555. convertLinkToAria(link, filename, ua) {
  1556. filename = base.fixFilename(filename);
  1557. return encodeURIComponent(`aria2c "${link}" --out "${filename}"`);
  1558. },
  1559.  
  1560. convertLinkToBC(link, filename, ua) {
  1561. let bc = `AA/${encodeURIComponent(filename)}/?url=${encodeURIComponent(link)}ZZ`;
  1562. return encodeURIComponent(`bc://http/${base.e(bc)}`);
  1563. },
  1564.  
  1565. convertLinkToCurl(link, filename, ua) {
  1566. let terminal = base.getValue('setting_terminal_type');
  1567. filename = base.fixFilename(filename);
  1568. return encodeURIComponent(`${terminal !== 'wp' ? 'curl' : 'curl.exe'} -L -C - "${link}" -o "${filename}"`);
  1569. },
  1570.  
  1571. addPageListener() {
  1572. doc.on('click', '.pl-button-mode', (e) => {
  1573. mode = e.target.dataset.mode;
  1574. Swal.showLoading();
  1575. this.getPCSLink();
  1576. });
  1577. doc.on('click', '.listener-link-api', async (e) => {
  1578. e.preventDefault();
  1579. $('#downloadIframe').attr('src', e.currentTarget.dataset.link);
  1580. });
  1581. doc.on('click', '.listener-link-aria, .listener-copy-all', (e) => {
  1582. e.preventDefault();
  1583. base.setClipboard(decodeURIComponent(e.target.dataset.link));
  1584. $(e.target).text('复制成功,快去粘贴吧!').animate({opacity: '0.5'}, "slow");
  1585. });
  1586. doc.on('click', '.listener-link-rpc', async (e) => {
  1587. let target = $(e.currentTarget);
  1588. target.find('.icon').remove();
  1589. target.find('.pl-loading').remove();
  1590. target.prepend(base.createLoading());
  1591. let res = await this.sendLinkToRPC(e.currentTarget.dataset.filename, e.currentTarget.dataset.link);
  1592. if (res === 'success') {
  1593. $('.listener-rpc-task').show();
  1594. target.removeClass('pl-btn-danger').html('发送成功,快去看看吧!').animate({opacity: '0.5'}, "slow");
  1595. } else {
  1596. target.addClass('pl-btn-danger').text('发送失败,请检查您的RPC配置信息!').animate({opacity: '0.5'}, "slow");
  1597. }
  1598. });
  1599. doc.on('click', '.listener-send-rpc', (e) => {
  1600. $('.listener-link-rpc').click();
  1601. $(e.target).text('发送完成,发送结果见上方按钮!').animate({opacity: '0.5'}, "slow");
  1602. });
  1603. doc.on('click', '.listener-open-setting', () => {
  1604. base.showSetting();
  1605. });
  1606. doc.on('click', '.listener-open-updatelog', () => {
  1607. base.showUpdateLog();
  1608. });
  1609. doc.on('click', '.listener-rpc-task', () => {
  1610. let rpc = JSON.stringify({
  1611. domain: base.getValue('setting_rpc_domain'),
  1612. port: base.getValue('setting_rpc_port'),
  1613. }), url = `${pan.d}/?rpc=${base.e(rpc)}#${base.getValue('setting_rpc_token')}`;
  1614. GM_openInTab(url, {active: true});
  1615. });
  1616. },
  1617.  
  1618. addButton() {
  1619. if (!pt) return;
  1620. let $toolWrap;
  1621. let $button = $(`<div class="tianyi-button pl-button">下载助手<ul class="pl-dropdown-menu" style="top: 26px;"><li class="pl-dropdown-menu-item pl-button-mode" data-mode="api">API下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="aria" >Aria下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="rpc">RPC下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="curl">cURL下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="bc" >BC下载</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-setting">助手设置</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-updatelog">更新日志</li></ul></div>`);
  1622. if (pt === 'home') {
  1623. let ins = setInterval(() => {
  1624. $toolWrap = $(pan.btn.home);
  1625. if ($toolWrap.length > 0) {
  1626. $toolWrap.prepend($button);
  1627. clearInterval(ins);
  1628. }
  1629. }, 50);
  1630. }
  1631. if (pt === 'share') {
  1632. let ins = setInterval(() => {
  1633. $toolWrap = $(pan.btn.share);
  1634. if ($toolWrap.length > 0) {
  1635. $toolWrap.prepend($button);
  1636. clearInterval(ins);
  1637. }
  1638. }, 50);
  1639. }
  1640. base.createDownloadIframe();
  1641. this.addPageListener();
  1642. },
  1643.  
  1644. addInitButton() {
  1645. if (!pt) return;
  1646. let $toolWrap;
  1647. let $button = $(`<div class="tianyi-button pl-button-init">下载助手(未点亮)</div>`);
  1648. if (pt === 'home') {
  1649. let ins = setInterval(() => {
  1650. $toolWrap = $(pan.btn.home);
  1651. if ($toolWrap.length > 0) {
  1652. $toolWrap.append($button);
  1653. clearInterval(ins);
  1654. }
  1655. }, 50);
  1656. }
  1657. if (pt === 'share') {
  1658. $button.css({'margin-right': '10px'});
  1659. let ins = setInterval(() => {
  1660. $toolWrap = $(pan.btn.share);
  1661. if ($toolWrap.length > 0) {
  1662. $toolWrap.prepend($button);
  1663. clearInterval(ins);
  1664. }
  1665. }, 50);
  1666. }
  1667. $button.click(() => base.initDialog());
  1668. },
  1669.  
  1670. async getToken() {
  1671. let res = await base.getFinalUrl(pan.pcs[1], {});
  1672. let accessToken = res.match(/accessToken=(\w+)/)?.[1];
  1673. accessToken && base.setStorage('accessToken', accessToken);
  1674. return accessToken;
  1675. },
  1676.  
  1677. async getFileUrlByOnce(item, index, token) {
  1678. try {
  1679. if (item.downloadUrl) return {
  1680. index,
  1681. downloadUrl: item.downloadUrl
  1682. };
  1683. let time = Date.now(),
  1684. fileId = item.fileId,
  1685. o = "AccessToken=" + token + "&Timestamp=" + time + "&fileId=" + fileId,
  1686. url = pan.pcs[2] + '?fileId=' + fileId;
  1687. if (item.shareId) {
  1688. o = "AccessToken=" + token + "&Timestamp=" + time + "&dt=1&fileId=" + fileId + "&shareId=" + item.shareId;
  1689. url += '&dt=1&shareId=' + item.shareId;
  1690. }
  1691. let sign = md5(o).toString();
  1692. let res = await base.get(url, {
  1693. "accept": "application/json;charset=UTF-8",
  1694. "sign-type": 1,
  1695. "accesstoken": token,
  1696. "timestamp": time,
  1697. "signature": sign
  1698. });
  1699. if (res.res_code === 0) {
  1700. return {
  1701. index,
  1702. downloadUrl: res.fileDownloadUrl
  1703. };
  1704. } else if (res.errorCode === 'InvalidSessionKey') {
  1705. return {
  1706. index,
  1707. downloadUrl: '提示:请先登录(不可用)网盘!'
  1708. };
  1709. } else if (res.res_code === 'ShareNotFoundFlatDir') {
  1710. return {
  1711. index,
  1712. downloadUrl: '提示:请先[转存]文件,👉前往[我的网盘]中下载!'
  1713. };
  1714. } else {
  1715. return {
  1716. index,
  1717. downloadUrl: '获取下载地址失败,请刷新重试!'
  1718. };
  1719. }
  1720. } catch (e) {
  1721. return {
  1722. index,
  1723. downloadUrl: '获取下载地址失败,请刷新重试!'
  1724. };
  1725. }
  1726. },
  1727.  
  1728. async getPCSLink() {
  1729. selectList = this.getSelectedList();
  1730. if (selectList.length === 0) {
  1731. return message.error('提示:请先勾选要下载的文件!');
  1732. }
  1733. if (this.isOnlyFolder()) {
  1734. return message.error('提示:请打开文件夹后勾选文件!');
  1735. }
  1736. let token = base.getStorage('accessToken') || await this.getToken();
  1737. if (!token) {
  1738. return message.error('提示:请先登录(不可用)网盘!');
  1739. }
  1740. let queue = [];
  1741. selectList.forEach((item, index) => {
  1742. queue.push(this.getFileUrlByOnce(item, index, token));
  1743. });
  1744.  
  1745. const res = await Promise.all(queue);
  1746. res.forEach(val => {
  1747. selectList[val.index].downloadUrl = val.downloadUrl;
  1748. });
  1749.  
  1750. let html = this.generateDom(selectList);
  1751. this.showMainDialog(pan[mode][0], html, pan[mode][1]);
  1752. },
  1753.  
  1754. generateDom(list) {
  1755. let content = '<div class="pl-main">';
  1756. let alinkAllText = '';
  1757. list.forEach((v, i) => {
  1758. if (v.isFolder) return;
  1759. let filename = v.fileName;
  1760. let size = base.sizeFormat(v.size);
  1761. let dlink = v.downloadUrl;
  1762. if (mode === 'api') {
  1763. content += `<div class="pl-item">
  1764. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1765. <a class="pl-item-link listener-link-api" data-filename="${filename}" data-link="${dlink}" data-index="${i}">${dlink}</a>
  1766. </div>`;
  1767. }
  1768. if (mode === 'aria') {
  1769. let alink = this.convertLinkToAria(dlink, filename, navigator.userAgent);
  1770. if (typeof (alink) === 'object') {
  1771. content += `<div class="pl-item">
  1772. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1773. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  1774. } else {
  1775. content += `<div class="pl-item">
  1776. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1777. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1778. }
  1779. }
  1780. if (mode === 'rpc') {
  1781. content += `<div class="pl-item">
  1782. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1783. <button class="pl-item-link listener-link-rpc pl-btn-primary pl-btn-info" data-filename="${filename}" data-link="${dlink}"><em class="icon icon-device"></em><span style="margin-left: 5px;">推送到 RPC 下载器</span></button></div>`;
  1784. }
  1785. if (mode === 'curl') {
  1786. let alink = this.convertLinkToCurl(dlink, filename, navigator.userAgent);
  1787. if (typeof (alink) === 'object') {
  1788. content += `<div class="pl-item">
  1789. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1790. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  1791. } else {
  1792. content += `<div class="pl-item">
  1793. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1794. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1795. }
  1796. }
  1797. if (mode === 'bc') {
  1798. let alink = this.convertLinkToBC(dlink, filename, navigator.userAgent);
  1799. content += `<div class="pl-item">
  1800. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  1801. <a class="pl-item-link" href="${decodeURIComponent(alink)}" title="点击用比特彗星下载" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  1802. }
  1803. });
  1804. content += '</div>';
  1805. if (mode === 'aria')
  1806. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button></div>`;
  1807. if (mode === 'rpc') {
  1808. let rpc = base.getValue('setting_rpc_domain') + ':' + base.getValue('setting_rpc_port') + base.getValue('setting_rpc_path');
  1809. content += `<div class="pl-extra"><button class="pl-btn-primary listener-send-rpc">发送全部链接</button><button title="${rpc}" class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px">设置 RPC 参数(当前为:${rpc})</button><button class="pl-btn-primary pl-btn-success listener-rpc-task" style="margin-left: 10px;display: none">查看下载任务</button></div>`;
  1810. }
  1811. if (mode === 'curl')
  1812. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button><button class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px;">设置终端类型(当前为:${terminalType[base.getValue('setting_terminal_type')]})</button></div>`;
  1813. return content;
  1814. },
  1815.  
  1816. async sendLinkToRPC(filename, link) {
  1817. let rpc = {
  1818. domain: base.getValue('setting_rpc_domain'),
  1819. port: base.getValue('setting_rpc_port'),
  1820. path: base.getValue('setting_rpc_path'),
  1821. token: base.getValue('setting_rpc_token'),
  1822. dir: base.getValue('setting_rpc_dir'),
  1823. };
  1824.  
  1825. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  1826. let rpcData = {
  1827. id: new Date().getTime(),
  1828. jsonrpc: '2.0',
  1829. method: 'aria2.addUri',
  1830. params: [`token:${rpc.token}`, [link], {
  1831. dir: rpc.dir,
  1832. out: filename,
  1833. header: []
  1834. }]
  1835. };
  1836. try {
  1837. let res = await base.post(url, rpcData, {}, '');
  1838. if (res.result) return 'success';
  1839. return 'fail';
  1840. } catch (e) {
  1841. return 'fail';
  1842. }
  1843. },
  1844.  
  1845. getSelectedList() {
  1846. try {
  1847. return document.querySelector(".c-file-list").__vue__.selectedList;
  1848. } catch (e) {
  1849. return [document.querySelector(".info-detail").__vue__.fileDetail];
  1850. }
  1851. },
  1852.  
  1853. detectPage() {
  1854. let path = location.pathname;
  1855. if (/^\/web\/main/.test(path)) return 'home';
  1856. if (/^\/web\/share/.test(path)) return 'share';
  1857. return '';
  1858. },
  1859.  
  1860. isOnlyFolder() {
  1861. for (let i = 0; i < selectList.length; i++) {
  1862. if (!selectList[i].isFolder) return false;
  1863. }
  1864. return true;
  1865. },
  1866.  
  1867. showMainDialog(title, html, footer) {
  1868. Swal.fire({
  1869. title,
  1870. html,
  1871. footer,
  1872. allowOutsideClick: false,
  1873. showCloseButton: true,
  1874. showConfirmButton: false,
  1875. position: 'top',
  1876. width,
  1877. padding: '15px 20px 5px',
  1878. customClass,
  1879. });
  1880. },
  1881.  
  1882. async initPanLinker() {
  1883. base.initDefaultConfig();
  1884. base.addPanLinkerStyle();
  1885. pt = this.detectPage();
  1886. if (base.getValue('setting_getuser_info') === 'yes') {
  1887. let res = await base.post
  1888. (`https://api.youxiaohou.com/config/tianyi?ver=${version}&a=${author}`, {}, {}, 'text');
  1889. pan = JSON.parse(base.d(res));
  1890. };
  1891. Object.freeze && Object.freeze(pan);
  1892. pan.num === base.getValue('setting_init_code') ? this.addButton() : this.addInitButton();
  1893. this.getToken();
  1894. base.createTip();
  1895. base.registerMenuCommand();
  1896. }
  1897. };
  1898.  
  1899. let xunlei = {
  1900.  
  1901. convertLinkToAria(link, filename, ua) {
  1902. filename = base.fixFilename(filename);
  1903. return encodeURIComponent(`aria2c "${link}" --out "${filename}"`);
  1904. },
  1905.  
  1906. convertLinkToBC(link, filename, ua) {
  1907. let bc = `AA/${encodeURIComponent(filename)}/?url=${encodeURIComponent(link)}ZZ`;
  1908. return encodeURIComponent(`bc://http/${base.e(bc)}`);
  1909. },
  1910.  
  1911. convertLinkToCurl(link, filename, ua) {
  1912. let terminal = base.getValue('setting_terminal_type');
  1913. filename = base.fixFilename(filename);
  1914. return encodeURIComponent(`${terminal !== 'wp' ? 'curl' : 'curl.exe'} -L -C - "${link}" -o "${filename}"`);
  1915. },
  1916.  
  1917. addPageListener() {
  1918. doc.on('click', '.pl-button-mode', (e) => {
  1919. mode = e.target.dataset.mode;
  1920. Swal.showLoading();
  1921. this.getPCSLink();
  1922. });
  1923. doc.on('click', '.listener-link-api', async (e) => {
  1924. e.preventDefault();
  1925. $('#downloadIframe').attr('src', e.currentTarget.dataset.link);
  1926. });
  1927. doc.on('click', '.listener-link-api-btn', async (e) => {
  1928. base.setClipboard(e.target.dataset.filename);
  1929. $(e.target).text('复制成功').animate({opacity: '0.5'}, "slow");
  1930. });
  1931. doc.on('click', '.listener-link-bc-btn', async (e) => {
  1932. let mirror = base.getMirrorList(e.target.dataset.dlink, pan.mirror);
  1933. base.setClipboard(mirror);
  1934. $(e.target).text('复制成功').animate({opacity: '0.5'}, "slow");
  1935. });
  1936. doc.on('click', '.listener-link-aria, .listener-copy-all', (e) => {
  1937. e.preventDefault();
  1938. base.setClipboard(decodeURIComponent(e.target.dataset.link));
  1939. $(e.target).text('复制成功,快去粘贴吧!').animate({opacity: '0.5'}, "slow");
  1940. });
  1941. doc.on('click', '.listener-link-rpc', async (e) => {
  1942. let target = $(e.currentTarget);
  1943. target.find('.icon').remove();
  1944. target.find('.pl-loading').remove();
  1945. target.prepend(base.createLoading());
  1946. let res = await this.sendLinkToRPC(e.currentTarget.dataset.filename, e.currentTarget.dataset.link);
  1947. if (res === 'success') {
  1948. $('.listener-rpc-task').show();
  1949. target.removeClass('pl-btn-danger').html('发送成功,快去看看吧!').animate({opacity: '0.5'}, "slow");
  1950. } else {
  1951. target.addClass('pl-btn-danger').text('发送失败,请检查您的RPC配置信息!').animate({opacity: '0.5'}, "slow");
  1952. }
  1953. });
  1954. doc.on('click', '.listener-send-rpc', (e) => {
  1955. $('.listener-link-rpc').click();
  1956. $(e.target).text('发送完成,发送结果见上方按钮!').animate({opacity: '0.5'}, "slow");
  1957. });
  1958. doc.on('click', '.listener-open-setting', () => {
  1959. base.showSetting();
  1960. });
  1961. doc.on('click', '.listener-open-updatelog', () => {
  1962. base.showUpdateLog();
  1963. });
  1964. doc.on('click', '.listener-rpc-task', () => {
  1965. let rpc = JSON.stringify({
  1966. domain: base.getValue('setting_rpc_domain'),
  1967. port: base.getValue('setting_rpc_port'),
  1968. }), url = `${pan.d}/?rpc=${base.e(rpc)}#${base.getValue('setting_rpc_token')}`;
  1969. GM_openInTab(url, {active: true});
  1970. });
  1971. },
  1972.  
  1973. addButton() {
  1974. if (!pt) return;
  1975. let $toolWrap;
  1976. let $button = $(`<div class="xunlei-button pl-button"><i class="xlpfont xlp-download"></i><span style="font-size: 13px;margin-left: 6px;">下载助手</span><ul class="pl-dropdown-menu" style="top: 34px;"><li class="pl-dropdown-menu-item pl-button-mode" data-mode="api">API下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="aria" >Aria下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="rpc">RPC下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="curl">cURL下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="bc" >BC下载</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-setting">助手设置</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-updatelog">更新日志</li></ul></div>`);
  1977. if (pt === 'home') {
  1978. let ins = setInterval(() => {
  1979. $toolWrap = $(pan.btn.home);
  1980. if ($toolWrap.length > 0) {
  1981. $toolWrap.prepend($button);
  1982. clearInterval(ins);
  1983. }
  1984. }, 50);
  1985. }
  1986. if (pt === 'share') {
  1987. $button.css({'margin-right': '10px'});
  1988. let ins = setInterval(() => {
  1989. $toolWrap = $(pan.btn.share);
  1990. if ($toolWrap.length > 0) {
  1991. $toolWrap.prepend($button);
  1992. clearInterval(ins);
  1993. }
  1994. }, 50);
  1995. }
  1996. base.createDownloadIframe();
  1997. this.addPageListener();
  1998. },
  1999.  
  2000. addInitButton() {
  2001. if (!pt) return;
  2002. let $toolWrap;
  2003. let $button = $(`<div class="xunlei-button pl-button-init"><i class="xlpfont xlp-download"></i><span style="font-size: 13px;margin-left: 6px;">下载助手(未点亮)</span></div>`);
  2004. if (pt === 'home') {
  2005. let ins = setInterval(() => {
  2006. $toolWrap = $(pan.btn.home);
  2007. if ($toolWrap.length > 0) {
  2008. $toolWrap.append($button);
  2009. clearInterval(ins);
  2010. }
  2011. }, 50);
  2012. }
  2013. if (pt === 'share') {
  2014. $button.css({'margin-right': '10px'});
  2015. let ins = setInterval(() => {
  2016. $toolWrap = $(pan.btn.share);
  2017. if ($toolWrap.length > 0) {
  2018. $toolWrap.prepend($button);
  2019. clearInterval(ins);
  2020. }
  2021. }, 50);
  2022. }
  2023. $button.click(() => base.initDialog());
  2024. },
  2025.  
  2026. getToken() {
  2027. let credentials = {}, captcha = {};
  2028. for (let i = 0; i < localStorage.length; i++) {
  2029. if (/^credentials_/.test(localStorage.key(i))) {
  2030. credentials = base.getStorage(localStorage.key(i));
  2031. base.setStorage('');
  2032. }
  2033. if (/^captcha_[\w]{16}/.test(localStorage.key(i))) {
  2034. captcha = base.getStorage(localStorage.key(i));
  2035. }
  2036. }
  2037. let deviceid = /(\w{32})/.exec(base.getStorage('deviceid').split(','))[0];
  2038. let token = {
  2039. credentials,
  2040. captcha,
  2041. deviceid
  2042. };
  2043. return token;
  2044. },
  2045.  
  2046. async getFileUrlByOnce(item, index, token) {
  2047. try {
  2048. if (item.downloadUrl) return {
  2049. index,
  2050. downloadUrl: item.downloadUrl
  2051. };
  2052. let res = await base.get(pan.pcs[0] + item.id, {
  2053. 'Authorization': `${token.credentials.token_type} ${token.credentials.access_token}`,
  2054. 'content-type': "application/json",
  2055. 'x-captcha-token': token.captcha.token,
  2056. 'x-device-id': token.deviceid,
  2057. });
  2058. if (res.web_content_link) {
  2059. return {
  2060. index,
  2061. downloadUrl: res.web_content_link
  2062. };
  2063. } else {
  2064. return {
  2065. index,
  2066. downloadUrl: '获取下载地址失败,请刷新重试!'
  2067. };
  2068. }
  2069. } catch (e) {
  2070. return message.error('提示:请先登录(不可用)网盘后刷新页面!');
  2071. }
  2072. },
  2073.  
  2074. async getPCSLink() {
  2075. selectList = this.getSelectedList();
  2076. if (selectList.length === 0) {
  2077. return message.error('提示:请先勾选要下载的文件!');
  2078. }
  2079. if (this.isOnlyFolder()) {
  2080. return message.error('提示:请打开文件夹后勾选文件!');
  2081. }
  2082. if (pt === 'home') {
  2083. let queue = [];
  2084. let token = this.getToken();
  2085. selectList.forEach((item, index) => {
  2086. queue.push(this.getFileUrlByOnce(item, index, token));
  2087. });
  2088. const res = await Promise.all(queue);
  2089. res.forEach(val => {
  2090. selectList[val.index].downloadUrl = val.downloadUrl;
  2091. });
  2092. } else {
  2093. return message.error('提示:请转存到自己网盘后去网盘主页下载!');
  2094. }
  2095. let html = this.generateDom(selectList);
  2096. this.showMainDialog(pan[mode][0], html, pan[mode][1]);
  2097.  
  2098. },
  2099.  
  2100. generateDom(list) {
  2101. let content = '<div class="pl-main">';
  2102. let alinkAllText = '';
  2103. list.forEach((v, i) => {
  2104. if (v.kind === 'drive#folder') return;
  2105. let filename = v.name;
  2106. let size = base.sizeFormat(+v.size);
  2107. let dlink = v.downloadUrl;
  2108. if (mode === 'api') {
  2109. content += `<div class="pl-item">
  2110. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2111. <a class="pl-item-link listener-link-api" data-filename="${filename}" data-link="${dlink}" data-index="${i}">${dlink}</a>
  2112. <div class="pl-item-btn listener-link-api-btn" data-filename="${filename}">复制文件名</div>
  2113. </div>`;
  2114. }
  2115. if (mode === 'aria') {
  2116. let alink = this.convertLinkToAria(dlink, filename, navigator.userAgent);
  2117. if (typeof (alink) === 'object') {
  2118. content += `<div class="pl-item">
  2119. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2120. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  2121. } else {
  2122. alinkAllText += alink + '\r\n';
  2123. content += `<div class="pl-item">
  2124. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2125. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  2126. }
  2127. }
  2128. if (mode === 'rpc') {
  2129. content += `<div class="pl-item">
  2130. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2131. <button class="pl-item-link listener-link-rpc pl-btn-primary pl-btn-info" data-filename="${filename}" data-link="${dlink}"><em class="icon icon-device"></em><span style="margin-left: 5px;">推送到 RPC 下载器</span></button></div>`;
  2132. }
  2133. if (mode === 'curl') {
  2134. let alink = this.convertLinkToCurl(dlink, filename, navigator.userAgent);
  2135. if (typeof (alink) === 'object') {
  2136. content += `<div class="pl-item">
  2137. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2138. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  2139. } else {
  2140. alinkAllText += alink + '\r\n';
  2141. content += `<div class="pl-item">
  2142. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2143. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  2144. }
  2145. }
  2146. if (mode === 'bc') {
  2147. let alink = this.convertLinkToBC(dlink, filename, navigator.userAgent);
  2148. content += `<div class="pl-item">
  2149. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2150. <a class="pl-item-link" href="${decodeURIComponent(alink)}" title="点击用比特彗星下载" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a>
  2151. <div class="pl-item-btn listener-link-bc-btn" data-dlink="${dlink}">复制镜像地址</div>
  2152. </div>`;
  2153. }
  2154. });
  2155. content += '</div>';
  2156. if (mode === 'aria')
  2157. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button></div>`;
  2158. if (mode === 'rpc') {
  2159. let rpc = base.getValue('setting_rpc_domain') + ':' + base.getValue('setting_rpc_port') + base.getValue('setting_rpc_path');
  2160. content += `<div class="pl-extra"><button class="pl-btn-primary listener-send-rpc">发送全部链接</button><button title="${rpc}" class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px">设置 RPC 参数(当前为:${rpc})</button><button class="pl-btn-primary pl-btn-success listener-rpc-task" style="margin-left: 10px;display: none">查看下载任务</button></div>`;
  2161. }
  2162. if (mode === 'curl')
  2163. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button><button class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px;">设置终端类型(当前为:${terminalType[base.getValue('setting_terminal_type')]})</button></div>`;
  2164. return content;
  2165. },
  2166.  
  2167. async sendLinkToRPC(filename, link) {
  2168. let rpc = {
  2169. domain: base.getValue('setting_rpc_domain'),
  2170. port: base.getValue('setting_rpc_port'),
  2171. path: base.getValue('setting_rpc_path'),
  2172. token: base.getValue('setting_rpc_token'),
  2173. dir: base.getValue('setting_rpc_dir'),
  2174. };
  2175.  
  2176. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  2177. let rpcData = {
  2178. id: new Date().getTime(),
  2179. jsonrpc: '2.0',
  2180. method: 'aria2.addUri',
  2181. params: [`token:${rpc.token}`, [link], {
  2182. dir: rpc.dir,
  2183. out: filename,
  2184. header: []
  2185. }]
  2186. };
  2187. try {
  2188. let res = await base.post(url, rpcData, {}, '');
  2189. if (res.result) return 'success';
  2190. return 'fail';
  2191. } catch (e) {
  2192. return 'fail';
  2193. }
  2194. },
  2195.  
  2196. getSelectedList() {
  2197. try {
  2198. let doms = document.querySelectorAll('.pan-list-item');
  2199. let selectedList = [];
  2200. for (let dom of doms) {
  2201. let domVue = dom.__vue__;
  2202. if (domVue.selected.includes(domVue.info.id)) {
  2203. selectedList.push(domVue.info);
  2204. }
  2205. }
  2206. return selectedList;
  2207. } catch (e) {
  2208. return [];
  2209. }
  2210. },
  2211.  
  2212. detectPage() {
  2213. let path = location.pathname;
  2214. if (/^\/$/.test(path)) return 'home';
  2215. if (/^\/(s|share)\//.test(path)) return 'share';
  2216. return '';
  2217. },
  2218.  
  2219. isOnlyFolder() {
  2220. for (let i = 0; i < selectList.length; i++) {
  2221. if (selectList[i].kind === 'drive#file') return false;
  2222. }
  2223. return true;
  2224. },
  2225.  
  2226. showMainDialog(title, html, footer) {
  2227. Swal.fire({
  2228. title,
  2229. html,
  2230. footer,
  2231. allowOutsideClick: false,
  2232. showCloseButton: true,
  2233. showConfirmButton: false,
  2234. position: 'top',
  2235. width,
  2236. padding: '15px 20px 5px',
  2237. customClass,
  2238. });
  2239. },
  2240.  
  2241. async initPanLinker() {
  2242. base.initDefaultConfig();
  2243. base.addPanLinkerStyle();
  2244. pt = this.detectPage();
  2245. if (base.getValue('setting_getuser_info') === 'yes') {
  2246. let res = await base.post
  2247. (`https://api.youxiaohou.com/config/xunlei?ver=${version}&a=${author}`, {}, {}, 'text');
  2248. pan = JSON.parse(base.d(res));
  2249. };
  2250. Object.freeze && Object.freeze(pan);
  2251. pan.num === base.getValue('setting_init_code') ? this.addButton() : this.addInitButton();
  2252. base.createTip();
  2253. base.registerMenuCommand();
  2254. }
  2255. };
  2256.  
  2257. let quark = {
  2258.  
  2259. convertLinkToAria(link, filename, ua) {
  2260. filename = base.fixFilename(filename);
  2261. return encodeURIComponent(`aria2c "${link}" --out "${filename}" --header "Cookie: ${document.cookie}"`);
  2262. },
  2263.  
  2264. convertLinkToBC(link, filename, ua) {
  2265. let bc = `AA/${encodeURIComponent(filename)}/?url=${encodeURIComponent(link)}&cookie=${encodeURIComponent(document.cookie)}ZZ`;
  2266. return encodeURIComponent(`bc://http/${base.e(bc)}`);
  2267. },
  2268.  
  2269. convertLinkToCurl(link, filename, ua) {
  2270. let terminal = base.getValue('setting_terminal_type');
  2271. filename = base.fixFilename(filename);
  2272. return encodeURIComponent(`${terminal !== 'wp' ? 'curl' : 'curl.exe'} -L -C - "${link}" -o "${filename}" -b "${document.cookie}"`);
  2273. },
  2274.  
  2275. addPageListener() {
  2276. window.addEventListener('hashchange', async (e) => {
  2277. let home = 'https://pan.quark.cn/list#/', all = 'https://pan.quark.cn/list#/list/all';
  2278. if (e.oldURL === home && e.newURL === all ) return;
  2279. await base.sleep(150);
  2280. if ($('.quark-button').length > 0) return;
  2281. });
  2282. window.addEventListener('hashchange', (event)=>{
  2283. pan.num === base.getValue('setting_init_code') ? this.addButton() : this.addInitButton();
  2284.  
  2285. })
  2286. doc.on('click', '.pl-button-mode', (e) => {
  2287. mode = e.target.dataset.mode;
  2288. Swal.showLoading();
  2289. this.getPCSLink();
  2290. });
  2291. doc.on('click', '.listener-link-api', async (e) => {
  2292. e.preventDefault();
  2293. $('#downloadIframe').attr('src', e.currentTarget.dataset.link);
  2294. });
  2295. doc.on('click', '.listener-link-aria, .listener-copy-all', (e) => {
  2296. e.preventDefault();
  2297. base.setClipboard(decodeURIComponent(e.target.dataset.link));
  2298. $(e.target).text('复制成功,快去粘贴吧!').animate({opacity: '0.5'}, "slow");
  2299. });
  2300. doc.on('click', '.listener-link-rpc', async (e) => {
  2301. let target = $(e.currentTarget);
  2302. target.find('.icon').remove();
  2303. target.find('.pl-loading').remove();
  2304. target.prepend(base.createLoading());
  2305. let res = await this.sendLinkToRPC(e.currentTarget.dataset.filename, e.currentTarget.dataset.link);
  2306. if (res === 'success') {
  2307. $('.listener-rpc-task').show();
  2308. target.removeClass('pl-btn-danger').html('发送成功,快去看看吧!').animate({opacity: '0.5'}, "slow");
  2309. } else {
  2310. target.addClass('pl-btn-danger').text('发送失败,请检查您的RPC配置信息!').animate({opacity: '0.5'}, "slow");
  2311. }
  2312. });
  2313. doc.on('click', '.listener-send-rpc', (e) => {
  2314. $('.listener-link-rpc').click();
  2315. $(e.target).text('发送完成,发送结果见上方按钮!').animate({opacity: '0.5'}, "slow");
  2316. });
  2317. doc.on('click', '.listener-open-setting', () => {
  2318. base.showSetting();
  2319. });
  2320. doc.on('click', '.listener-open-updatelog', () => {
  2321. base.showUpdateLog();
  2322. });
  2323. doc.on('click', '.listener-rpc-task', () => {
  2324. let rpc = JSON.stringify({
  2325. domain: base.getValue('setting_rpc_domain'),
  2326. port: base.getValue('setting_rpc_port'),
  2327. }), url = `${pan.d}/?rpc=${base.e(rpc)}#${base.getValue('setting_rpc_token')}`;
  2328. GM_openInTab(url, {active: true});
  2329. });
  2330. },
  2331.  
  2332. addButton() {
  2333. if ($("#quark-button")){
  2334. $("#quark-button").remove();
  2335. };
  2336. if (!pt) return;
  2337. let $toolWrap;
  2338. let $button = $(`<div id="quark-button" class="file-info_r quark-button pl-button"><svg width="22" height="22" xmlns="http://www.w3.org/2000/svg"><g fill="none" fill-rule="evenodd" stroke="#FFFFFF" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M9 12l2 2 2-2z"/><path d="M14 8h1.553c.85 0 1.16.093 1.47.267.311.174.556.43.722.756.166.326.255.65.255 1.54v4.873c0 .892-.089 1.215-.255 1.54-.166.327-.41.583-.722.757-.31.174-.62.267-1.47.267H6.447c-.85 0-1.16-.093-1.47-.267a1.778 1.778 0 01-.722-.756c-.166-.326-.255-.65-.255-1.54v-4.873c0-.892.089-1.215.255-1.54.166-.327.41-.583.722-.757.31-.174.62-.267 1.47-.267H11"/><path stroke-linecap="round" stroke-linejoin="round" d="M11 3v10"/></g></svg>下载助手<ul class="pl-dropdown-menu"><li class="pl-dropdown-menu-item pl-button-mode" data-mode="api">API下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="aria" >Aria下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="rpc">RPC下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="curl">cURL下载</li><li class="pl-dropdown-menu-item pl-button-mode" data-mode="bc" >BC下载</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-setting">助手设置</li><li class="pl-dropdown-menu-item pl-button-mode listener-open-updatelog">更新日志</li></ul></div>`);
  2339. $button.css({"margin-right":"10px","background-color":color});
  2340. if (pt === 'home') {
  2341. let ins = setInterval(() => {
  2342. $toolWrap = $(pan.btn.home);
  2343. if ($toolWrap.length > 0) {
  2344. $toolWrap.prepend($button);
  2345. clearInterval(ins);
  2346. }
  2347. }, 50);
  2348. }
  2349. if (pt === 'share') {
  2350. $button.css({"margin-right":"10px","background-color":color});
  2351. let ins = setInterval(() => {
  2352. $toolWrap = $(pan.btn.share);
  2353. if ($toolWrap.length > 0) {
  2354. $toolWrap.prepend($button);
  2355. clearInterval(ins);
  2356. }
  2357. }, 50);
  2358. }
  2359. },
  2360.  
  2361. addInitButton() {
  2362. $("#pl-button-init").remove();
  2363. if (!pt) return;
  2364. let $toolWrap;
  2365. let $button = $(`<div id="quark-button" class="file-info_r quark-button pl-button-init"><svg width="22" height="22" xmlns="http://www.w3.org/2000/svg"><g fill="none" fill-rule="evenodd" stroke="#FFFFFF" stroke-width="2"><path stroke-linecap="round" stroke-linejoin="round" d="M9 12l2 2 2-2z"/><path d="M14 8h1.553c.85 0 1.16.093 1.47.267.311.174.556.43.722.756.166.326.255.65.255 1.54v4.873c0 .892-.089 1.215-.255 1.54-.166.327-.41.583-.722.757-.31.174-.62.267-1.47.267H6.447c-.85 0-1.16-.093-1.47-.267a1.778 1.778 0 01-.722-.756c-.166-.326-.255-.65-.255-1.54v-4.873c0-.892.089-1.215.255-1.54.166-.327.41-.583.722-.757.31-.174.62-.267 1.47-.267H11"/><path stroke-linecap="round" stroke-linejoin="round" d="M11 3v10"/></g></svg>下载助手(未点亮)</div>`);
  2366. $button.css({"margin-right":"10px","background-color":color});
  2367. if (pt === 'home') {
  2368. let ins = setInterval(() => {
  2369. $toolWrap = $(pan.btn.home);
  2370. if ($toolWrap.length > 0) {
  2371. $toolWrap.prepend($button);
  2372. clearInterval(ins);
  2373. }
  2374. }, 50);
  2375. }
  2376. if (pt === 'share') {
  2377. $button.css({'margin-right': '10px','width': '160px',"background-color":color});
  2378. let ins = setInterval(() => {
  2379. $toolWrap = $(pan.btn.share);
  2380. if ($toolWrap.length > 0) {
  2381. $toolWrap.prepend($button);
  2382. clearInterval(ins);
  2383. }
  2384. }, 50);
  2385. }
  2386. $button.click(() => base.initDialog());
  2387. },
  2388.  
  2389. async getPCSLink() {
  2390. selectList = this.getSelectedList();
  2391. if (selectList.length === 0) {
  2392. return message.error('提示:请先勾选要下载的文件!');
  2393. }
  2394. if (this.isOnlyFolder()) {
  2395. return message.error('提示:请打开文件夹后勾选文件!');
  2396. }
  2397. let fids = [];
  2398. selectList.forEach(val => {
  2399. fids.push(val.fid);
  2400. });
  2401. if (pt === 'home') {
  2402. let res = await base.post(pan.pcs[0], {
  2403. "fids": fids
  2404. }, {"content-type": "application/json;charset=utf-8"});
  2405. if (res.code === 31001) {
  2406. return message.error('提示:请先登录(不可用)网盘!');
  2407. }
  2408. if (res.code !== 0) {
  2409. return message.error('提示:获取链接失败!');
  2410. }
  2411. let html = this.generateDom(res.data);
  2412. this.showMainDialog(pan[mode][0], html, pan[mode][1]);
  2413. } else {
  2414. message.error('提示:请保存到自己网盘后去网盘主页下载!');
  2415. await base.sleep(1000);
  2416. document.querySelector('.file-info_r').click();
  2417. return
  2418. }
  2419. },
  2420.  
  2421. generateDom(list) {
  2422. let content = '<div class="pl-main">';
  2423. let alinkAllText = '';
  2424. list.forEach((v, i) => {
  2425. if (v.file === false) return;
  2426. let filename = v.file_name;
  2427. let fid = v.fid;
  2428. let size = base.sizeFormat(v.size);
  2429. let dlink = v.download_url;
  2430. if (mode === 'api') {
  2431. content += `<div class="pl-item">
  2432. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2433. <a class="pl-item-link listener-link-api" data-fid="${fid}" data-filename="${filename}" data-link="${dlink}" data-index="${i}">${dlink}</a>
  2434. </div>`;
  2435. }
  2436. if (mode === 'aria') {
  2437. let alink = this.convertLinkToAria(dlink, filename, navigator.userAgent);
  2438. if (typeof (alink) === 'object') {
  2439. content += `<div class="pl-item">
  2440. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2441. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  2442. } else {
  2443. alinkAllText += alink + '\r\n';
  2444. content += `<div class="pl-item">
  2445. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2446. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制aria2c链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  2447. }
  2448. }
  2449. if (mode === 'rpc') {
  2450. content += `<div class="pl-item">
  2451. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2452. <button class="pl-item-link listener-link-rpc pl-btn-primary pl-btn-info" data-filename="${filename}" data-link="${dlink}"><em class="icon icon-device"></em><span style="margin-left: 5px;">推送到 RPC 下载器</span></button></div>`;
  2453. }
  2454. if (mode === 'curl') {
  2455. let alink = this.convertLinkToCurl(dlink, filename, navigator.userAgent);
  2456. if (typeof (alink) === 'object') {
  2457. content += `<div class="pl-item">
  2458. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2459. <a class="pl-item-link" target="_blank" href="${alink.link}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink.link}">${decodeURIComponent(alink.text)}</a> </div>`;
  2460. } else {
  2461. alinkAllText += alink + '\r\n';
  2462. content += `<div class="pl-item">
  2463. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2464. <a class="pl-item-link listener-link-aria" href="${alink}" title="点击复制curl链接" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  2465. }
  2466. }
  2467. if (mode === 'bc') {
  2468. let alink = this.convertLinkToBC(dlink, filename, navigator.userAgent);
  2469. content += `<div class="pl-item">
  2470. <div class="pl-item-name listener-tip" data-size="${size}">${filename}</div>
  2471. <a class="pl-item-link" href="${decodeURIComponent(alink)}" title="点击用比特彗星下载" data-filename="${filename}" data-link="${alink}">${decodeURIComponent(alink)}</a> </div>`;
  2472. }
  2473. });
  2474. content += '</div>';
  2475. if (mode === 'aria')
  2476. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button></div>`;
  2477. if (mode === 'rpc') {
  2478. let rpc = base.getValue('setting_rpc_domain') + ':' + base.getValue('setting_rpc_port') + base.getValue('setting_rpc_path');
  2479. content += `<div class="pl-extra"><button class="pl-btn-primary listener-send-rpc">发送全部链接</button><button title="${rpc}" class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px">设置 RPC 参数(当前为:${rpc})</button><button class="pl-btn-primary pl-btn-success listener-rpc-task" style="margin-left: 10px;display: none">查看下载任务</button></div>`;
  2480. }
  2481. if (mode === 'curl')
  2482. content += `<div class="pl-extra"><button class="pl-btn-primary listener-copy-all" data-link="${alinkAllText}">复制全部链接</button><button class="pl-btn-primary pl-btn-warning listener-open-setting" style="margin-left: 10px;">设置终端类型(当前为:${terminalType[base.getValue('setting_terminal_type')]})</button></div>`;
  2483. return content;
  2484. },
  2485.  
  2486. async sendLinkToRPC(filename, link) {
  2487. let rpc = {
  2488. domain: base.getValue('setting_rpc_domain'),
  2489. port: base.getValue('setting_rpc_port'),
  2490. path: base.getValue('setting_rpc_path'),
  2491. token: base.getValue('setting_rpc_token'),
  2492. dir: base.getValue('setting_rpc_dir'),
  2493. };
  2494.  
  2495. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  2496. let rpcData = {
  2497. id: new Date().getTime(),
  2498. jsonrpc: '2.0',
  2499. method: 'aria2.addUri',
  2500. params: [`token:${rpc.token}`, [link], {
  2501. dir: rpc.dir,
  2502. out: filename,
  2503. header: [`Cookie: ${document.cookie}`]
  2504. }]
  2505. };
  2506. try {
  2507. let res = await base.post(url, rpcData, {"Cookie": document.cookie}, '');
  2508. if (res.result) return 'success';
  2509. return 'fail';
  2510. } catch (e) {
  2511. return 'fail';
  2512. }
  2513. },
  2514.  
  2515. getSelectedList() {
  2516. try {
  2517. let selectedList = [];
  2518. let reactDom = document.getElementsByClassName('file-list')[0];
  2519. let reactObj = base.findReact(reactDom);
  2520. let props = reactObj.props;
  2521. if (props) {
  2522. let fileList = props.list || [];
  2523. let selectedKeys = props.selectedRowKeys || [];
  2524. fileList.forEach((val) => {
  2525. if (selectedKeys.includes(val.fid)) {
  2526. selectedList.push(val);
  2527. }
  2528. });
  2529. }
  2530. return selectedList;
  2531. } catch (e) {
  2532. return [];
  2533. }
  2534. },
  2535.  
  2536. detectPage() {
  2537. let path = location.pathname;
  2538. if (/^\/(list)/.test(path)) return 'home';
  2539. if (/^\/(s|share)\//.test(path)) return 'share';
  2540. return '';
  2541. },
  2542.  
  2543. isOnlyFolder() {
  2544. for (let i = 0; i < selectList.length; i++) {
  2545. if (selectList[i].file) return false;
  2546. }
  2547. return true;
  2548. },
  2549.  
  2550. showMainDialog(title, html, footer) {
  2551. Swal.fire({
  2552. title,
  2553. html,
  2554. footer,
  2555. allowOutsideClick: false,
  2556. showCloseButton: true,
  2557. showConfirmButton: false,
  2558. position: 'top',
  2559. width,
  2560. padding: '15px 20px 5px',
  2561. customClass,
  2562. });
  2563. },
  2564.  
  2565. async initPanLinker() {
  2566. base.initDefaultConfig();
  2567. base.addPanLinkerStyle();
  2568. pt = this.detectPage();
  2569. if (base.getValue('setting_getuser_info') === 'yes') {
  2570. let res = await base.post
  2571. (`https://api.youxiaohou.com/config/quark?ver=${version}&a=${author}`, {}, {}, 'text');
  2572. pan = JSON.parse(base.d(res));
  2573. };
  2574. Object.freeze && Object.freeze(pan);
  2575. pan.num === base.getValue('setting_init_code') ? this.addButton() : this.addInitButton();
  2576. this.addPageListener();
  2577. base.createTip();
  2578. base.createDownloadIframe();
  2579. base.registerMenuCommand();
  2580. },
  2581. };
  2582.  
  2583. let main = {
  2584. init() {
  2585. if (/(pan|yun).baidu.com/.test(location.host)) {
  2586. baidu.initPanLinker();
  2587. }
  2588. if (/www.aliyundrive.com/.test(location.host)) {
  2589. ali.initPanLinker();
  2590. }
  2591. if (/cloud.189.cn/.test(location.host)) {
  2592. tianyi.initPanLinker();
  2593. }
  2594. if (/pan.xunlei.com/.test(location.host)) {
  2595. xunlei.initPanLinker();
  2596. }
  2597. if (/pan.quark.cn/.test(location.host)) {
  2598. quark.initPanLinker();
  2599. }
  2600. }
  2601. };
  2602.  
  2603. main.init();
  2604. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址