LinkSwift

(。>ᴗ•)✧《也许同类型中最好用?》系列 - 一个基于 JavaScript 的网盘文件下载地址获取工具✨,基于【网盘直链下载助手】修改 | 支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼云盘 / 迅雷云盘 / 夸克网盘 / UC网盘 / 123云盘 八大网盘 | 开源・自用・去广 | 改界面・添功能・修Bug | 既超越原版,亦是同类中最好用版本!👋

Instalar este script¿?
Script recomendado por el autor

Puede que también te guste (改)百度网盘会员青春版.

Instalar este script
  1. // ==UserScript==
  2. // @name LinkSwift
  3. // @namespace github.com/hmjz100
  4. // @version 1.1.1.7
  5. // @author Hmjz100、油小猴
  6. // @icon 
  7. // @description (。>ᴗ•)✧《也许同类型中最好用?》系列 - 一个基于 JavaScript 的网盘文件下载地址获取工具✨,基于【网盘直链下载助手】修改 | 支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼云盘 / 迅雷云盘 / 夸克网盘 / UC网盘 / 123云盘 八大网盘 | 开源・自用・去广 | 改界面・添功能・修Bug | 既超越原版,亦是同类中最好用版本!👋
  8. // @description:zh-CN (。>ᴗ•)✧《也许同类型中最好用?》系列 - 一个基于 JavaScript 的网盘文件下载地址获取工具✨,基于【网盘直链下载助手】修改 | 支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼云盘 / 迅雷云盘 / 夸克网盘 / UC网盘 / 123云盘 八大网盘 | 开源・自用・去广 | 改界面・添功能・修Bug | 既超越原版,亦是同类中最好用版本!👋
  9. // @description:zh-TW (。>ᴗ•)✧《也許同類型中最好用?》系列 - 一個基於 JavaScript 的網盤檔案下載地址獲取工具✨,基於【網盤直鏈下載助手】改編 | 支援 百度網盤 / 阿里雲盤 / 中國移動雲盤 / 天翼雲盤 / 迅雷雲盤 / 夸克網盤 / UC網盤 / 123雲盤 八大平台 | 開源・自用・除廣 | 改介面・擴功能・修Bug | 既超越原版,亦是同類中最好用版本!👋
  10. // @description:zh-HK (。>ᴗ•)✧《也許同類型中最好用?》系列 - 一個基於 JavaScript 的網盤檔案下載地址獲取工具✨,基於【網盤直鏈下載助手】改編 | 支援 百度網盤 / 阿里雲盤 / 中國移動雲盤 / 天翼雲盤 / 迅雷雲盤 / 夸克網盤 / UC網盤 / 123雲盤 八大平台 | 開源・自用・除廣 | 改介面・擴功能・修Bug | 既超越原版,亦是同類中最好用版本!👋
  11. // @license AGPL-3.0-or-later
  12. // @homepage https://github.com/hmjz100/LinkSwift/
  13. // @support https://github.com/hmjz100/LinkSwift/issues
  14. // @supportURL https://github.com/hmjz100/LinkSwift/issues
  15. // @require https://unpkg.com/jquery@3.6.0/dist/jquery.min.js
  16. // @require https://unpkg.com/sweetalert2@11.4.8/dist/sweetalert2.min.js
  17. // @resource SwalLigt https://unpkg.com/sweetalert2@11.4.8/dist/sweetalert2.min.css
  18. // @resource SwalDark https://unpkg.com/@sweetalert2/theme-dark@5.0.26/dark.min.css
  19. // @require https://unpkg.com/js-md5@0.7.3/build/md5.min.js
  20. // @run-at document-start
  21. // @match *://pan.baidu.com/disk/home*
  22. // @match *://yun.baidu.com/disk/home*
  23. // @match *://pan.baidu.com/disk/timeline*
  24. // @match *://yun.baidu.com/disk/timeline*
  25. // @match *://pan.baidu.com/disk/main*
  26. // @match *://yun.baidu.com/disk/main*
  27. // @match *://pan.baidu.com/youth/pan/main*
  28. // @match *://yun.baidu.com/youth/pan/main*
  29. // @match *://pan.baidu.com/disk/base*
  30. // @match *://yun.baidu.com/disk/base*
  31. // @match *://pan.baidu.com/disk/timeline*
  32. // @match *://yun.baidu.com/disk/timeline*
  33. // @match *://pan.baidu.com/pfile/*
  34. // @match *://yun.baidu.com/pfile/*
  35. // @match *://pan.baidu.com/s/*
  36. // @match *://pan.baidu.com/aipan/*
  37. // @match *://yun.baidu.com/s/*
  38. // @match *://yun.baidu.com/aipan/*
  39. // @match *://pan.baidu.com/share/*
  40. // @match *://yun.baidu.com/share/*
  41. // @match *://pan.baidu.com/embed/*
  42. // @match *://yun.baidu.com/embed/*
  43. // @match *://openapi.baidu.com/*
  44. // @match *://www.aliyundrive.com/s/*
  45. // @match *://www.aliyundrive.com/drive*
  46. // @match *://www.alipan.com/s/*
  47. // @match *://www.alipan.com/drive*
  48. // @match *://cloud.189.cn/web/*
  49. // @match *://pan.xunlei.com/*
  50. // @match *://pan.quark.cn/*
  51. // @match *://drive.uc.cn/*
  52. // @match *://yun.139.com/*
  53. // @match *://caiyun.139.com/*
  54. // @match *://*.123pan.com/*
  55. // @match *://*.123pan.cn/*
  56. // @match *://*.123684.com/*
  57. // @match *://*.123865.com/*
  58. // @match *://*.123952.com/*
  59. // @match *://*.123912.com/*
  60. // @connect *
  61. // @connect localhost
  62. // @connect baidu.com
  63. // @connect baidupcs.com
  64. // @connect aliyundrive.com
  65. // @connect aliyundrive.net
  66. // @connect alipan.com
  67. // @connect alicloudccp.com
  68. // @connect aliyundrive.cloud
  69. // @connect 139.com
  70. // @connect cmecloud.cn
  71. // @connect 189.cn
  72. // @connect xunlei.com
  73. // @connect quark.cn
  74. // @connect uc.cn
  75. // @connect 123pan.com
  76. // @connect 123pan.cn
  77. // @connect 123684.com
  78. // @connect 123865.com
  79. // @connect 123952.com
  80. // @connect 123912.com
  81. // @connect cjjd19.com
  82. // @grant unsafeWindow
  83. // @grant window.close
  84. // @grant GM_xmlhttpRequest
  85. // @grant GM_setClipboard
  86. // @grant GM_setValue
  87. // @grant GM_getValue
  88. // @grant GM_deleteValue
  89. // @grant GM_openInTab
  90. // @grant GM_registerMenuCommand
  91. // @grant GM_getResourceText
  92. // @compatible Chrome
  93. // @compatible Edge
  94. // @compatible Firefox
  95. // @compatible Safari
  96. // @compatible Opera
  97. // ==/UserScript==
  98. /**
  99. * @name LinkSwift
  100. * @template (改)网盘直链下载助手
  101. * @author 油小猴
  102. * @author hmjz100
  103. * @namespace github.com/hmjz100
  104. * @description 一个基于 JavaScript 盘的文件下载地址获取工具
  105. * 支持 百度网盘/阿里云盘/中国移动云盘/天翼云盘/迅雷云盘/夸克网盘/UC网盘/123云盘 八大网盘
  106. * @version 1.1.1.7
  107. * @license AGPL-3.0-or-later
  108. * @see {@link https://github.com/hmjz100/LinkSwift/ Github 仓库}
  109. */
  110. (function linkSwift() {
  111. // 严格模式,确保代码安全执行,不越界
  112. 'use strict';
  113. // unsafeWindow 检测,适用于 Via 这类无 unsafeWindow 的浏览器
  114. if (typeof (unsafeWindow) === 'undefined') window.unsafeWindow = window;
  115. /*
  116. 防止代码因其他原因被执行多次
  117. 代码出自 “Via 轻插件”,作者谷花泰
  118. */
  119. let key = encodeURIComponent('LinkSwift:主代码');
  120. if (window[key]) return;
  121. window[key] = true;
  122. /*
  123. 网盘直链下载助手
  124. 代码改自 “网盘直链下载助手”,作者油小猴
  125. 有增添新代码
  126. */
  127. /* 全局参数 */
  128. let mount = idontknow("LinkSwift");
  129. let info = {
  130. author: GM_info.script?.author,
  131. name: GM_info.script?.name,
  132. version: (GM_info.script?.version?.toString() || "1.1.1.7"),
  133. icon: (GM_info.script?.icon || ""),
  134. mhandler: GM_info.scriptHandler,
  135. mversion: GM_info.version,
  136. };
  137. let $doc = $(document);
  138. let temp = {
  139. pege: "",
  140. mode: [],
  141. links: [],
  142. color: "",
  143. progress: {},
  144. request: {},
  145. ins: {},
  146. idm: {},
  147. selectList: [],
  148. colored: false,
  149. swalDefault: {
  150. position: 'center',
  151. heightAuto: false,
  152. scrollbarPadding: false,
  153. confirmButtonText: `<svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"/></svg> 确认`,
  154. denyButtonText: `<svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg> 拒绝`,
  155. cancelButtonText: `<svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg> 取消`
  156. },
  157. terminalType: {
  158. wc: "Microsoft Windows 命令提示符",
  159. wp: "Microsoft Windows PowerShell",
  160. lt: "Linux 终端",
  161. ls: "Linux Shell",
  162. mt: "Apple MacOS 终端"
  163. }
  164. };
  165. /**
  166. * SweetAlert2 的 Toast 提示框基础配置
  167. * @author 油小猴
  168. * @author hmjz100
  169. * @description 创建一个全局通用的 Toast 提示框实例,支持自动关闭、鼠标悬停暂停、右上角弹出等特性。
  170. *
  171. * @type{Sweetalert2.Toast}
  172. */
  173. let toast = Swal.mixin({
  174. toast: true,
  175. position: 'top-end',
  176. showConfirmButton: false,
  177. timer: 3500,
  178. timerProgressBar: true,
  179. showCloseButton: true,
  180. didOpen: function (toast) {
  181. toast.addEventListener('mouseenter', () => {
  182. Swal.stopTimer();
  183. });
  184. toast.addEventListener('mouseleave', () => {
  185. Swal.resumeTimer();
  186. });
  187. }
  188. });
  189. /**
  190. * 消息提示工具类
  191. * @author 油小猴
  192. * @description 提供统一的提示信息展示方法,基于 SweetAlert2 的 Toast 实现;
  193. * 包含 success / error / warning / info / question 等类型。
  194. */
  195. let message = {
  196. success: function (text) {
  197. toast.fire({ title: text, icon: 'success' });
  198. },
  199. error: function (text) {
  200. toast.fire({ title: text, icon: 'error' });
  201. },
  202. warning: function (text) {
  203. toast.fire({ title: text, icon: 'warning' });
  204. },
  205. info: function (text) {
  206. toast.fire({ title: text, icon: 'info' });
  207. },
  208. question: function (text) {
  209. toast.fire({ title: text, icon: 'question' });
  210. }
  211. };
  212. /**
  213. * 基础配置集合
  214. * @author 油小猴
  215. * @author hmjz100
  216. */
  217. let config = {
  218. base: {
  219. num: "865746",
  220. license: "AGPL3",
  221. service: {
  222. account: "https://pic.rmb.bdstatic.com/bjh/8b9e14345b3cdf96aedac2f3971adcb02681.png"
  223. },
  224. dom: {
  225. footer: `o(≧▽≦)o 十分感谢您的支持!来给此项目一个 <a href="https://github.com/hmjz100/LinkSwift" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg>Star</a> 吧~`,
  226. button: {
  227. api: {
  228. title: "API 下载",
  229. footer: `<p>适用于 <a href="https://www.youxiaohou.com/zh-cn/idm.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>IDM</a>,<a href="https://www.youxiaohou.com/zh-cn/ndm.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>NDM</a> 以及浏览器自带下载</p>`
  230. },
  231. aria2: {
  232. title: "Aria2 下载",
  233. footer: `<p>RPC 适用于 <a href="https://www.youxiaohou.com/zh-cn/motrix.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>Motrix</a>,<a href="https://www.youxiaohou.com/download.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>Aria2 Tools</a>,<a href="https://www.youxiaohou.com/download.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>AriaNgGUI</a></p>
  234. <p>命令行适用于 <a href="https://www.youxiaohou.com/zh-cn/xdown.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>XDown</a> 及 <a href="https://www.youxiaohou.com/zh-cn/linux.html#linux-shell" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>Linux Shell 命令行</a></p>`
  235. },
  236. curl: {
  237. title: "cURL 下载",
  238. footer: `<p>适用于 <a href="https://www.youxiaohou.com/zh-cn/curl.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>WindowsLinuxMacOS 终端</a></p>`
  239. },
  240. bitcomet: {
  241. title: "比特彗星下载",
  242. footer: `<p>适用于 <a href="https://www.youxiaohou.com/zh-cn/bitcomet.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>比特彗星</a></p>`
  243. },
  244. abdm: {
  245. title: "ABDM 下载",
  246. footer: `<p>适用于 <a href="https://abdownloadmanager.com/" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg>AB Download Manager</a></p>`
  247. }
  248. },
  249. theme: [
  250. { color: '#09AAFF', name: '度盘|经典蓝' },
  251. { color: '#cc3235', name: '度盘|平安红' },
  252. { color: '#518c17', name: '度盘|盎然绿' },
  253. { color: '#ed944b', name: '度盘|周年橙' },
  254. { color: '#f969a5', name: '度盘|幸会粉' },
  255. { color: '#bca280', name: '度盘|午后棕' },
  256. { color: '#b673ab', name: '度盘|物语紫' },
  257. { color: '#574AB8', name: '度盘|星空紫' },
  258. { color: '#1d2327', name: 'OpenAI|默认黑' },
  259. { color: '#18a497', name: 'OpenAI|默认青' },
  260. { color: '#637dff', name: '度里叁|霞光紫' },
  261. { color: '#0d53ff', name: '夸克|极简蓝' },
  262. { color: '#3181f9', name: '移动|彩云蓝' },
  263. { color: '#f8d800', name: '果核|柠檬黄' },
  264. { color: '#0396ff', name: '果核|默认蓝' },
  265. { color: '#32ccbc', name: '果核|碧波绿' },
  266. { color: '#f6416c', name: '果核|玫瑰红' },
  267. { color: '#2271b1', name: '文派|默认蓝' },
  268. { color: '#59524c', name: '文派|咖啡灰' },
  269. { color: '#ff679a', name: '哔哩|少女粉' },
  270. { color: '#f44236', name: '哔哩|高能红' },
  271. { color: '#fec107', name: '哔哩|咸蛋黄' },
  272. { color: '#8bc24a', name: '哔哩|早苗绿' },
  273. { color: '#2594ed', name: '哔哩|宝石蓝' },
  274. { color: '#9c28b1', name: '哔哩|罗兰紫' }
  275. ]
  276. }
  277. },
  278. $baidu: {
  279. api: {
  280. ua: {
  281. downloadLink: "pan.baidu.com"
  282. },
  283. getAccessToken: "https://openapi.baidu.com/oauth/2.0/authorize?response_type=token&scope=basic,netdisk&client_id=IlLqBbU3GjQ0t46TRwFateTprHWl39zF&redirect_uri=oob&confirm_login=0",
  284. getLink: "https://pan.baidu.com/rest/2.0/xpan/multimedia?method=filemetas&dlink=1",
  285. getFiles: "https://pan.baidu.com/rest/2.0/xpan/file?method=list&showempty=1",
  286. getShareLink: "https://pan.baidu.com/api/sharedownload?channel=chunlei&clienttype=0&web=1&app_id=250528",
  287. getShareInfo: "https://pan.baidu.com/share/tplconfig?fields=sign,timestamp&channel=chunlei&web=1&app_id=250528&clienttype=0",
  288. getShareFiles: "https://pan.baidu.com/rest/2.0/xpan/share?method=list&showempty=1"
  289. },
  290. mount: {
  291. home: ".frame-main>div>div>div>div:has(.g-dropdown-button.g-new-create)",
  292. main: ".wp-s-agile-tool-bar__header",
  293. share: ".module-share-top-bar .x-button-box .g-dropdown-button.tools-more"
  294. },
  295. dom: {
  296. enhance: `+<br/>此方式下载有可能会被 IDM 捕获下载链接`,
  297. normal: `+<br/>不支持超过 50MB 的文件,若超过点击会没有反应<br/>此方式下载有可能会被 IDM 捕获下载链接`,
  298. copy: `不建议使用本功能,在百度网盘中单独复制链接并粘贴下载可能会导致服务器回报 403 错误<br/>如仍需使用,请搭配此用户代理进行下载:pan.baidu.com`
  299. }
  300. },
  301. $aliyun: {
  302. api: {
  303. getLink: "https://api.aliyundrive.com/v2/file/get_download_url",
  304. getShareLink: "https://api.aliyundrive.com/v2/file/get_share_link_download_url"
  305. },
  306. mount: {
  307. home: "[class^=\"header--\"]>[class^=\"actions--\"]",
  308. share: "[class^=\"banner--\"]>[class^=\"right--\"]",
  309. list: "[class^=\"node-list-table-view--\"]",
  310. grid: "[class^=\"node-list-grid-view--\"]",
  311. switch: "[class^=\"switch-wrapper--\"]"
  312. },
  313. dom: {
  314. enhance: `+<br/>此方式下载不会被 IDM 捕获下载链接`,
  315. normal: `+<br/>此方式下载有可能会被 IDM 捕获下载链接`,
  316. copy: `不建议使用本功能,在阿里云盘中单独复制链接并粘贴下载可能会导致服务器回报 403 错误`,
  317. filename: `阿里云盘于下载高峰期时可能不会显示文件名称,这时需要手动复制文件名称到下载工具中`
  318. }
  319. },
  320. $mcloud: {
  321. api: {
  322. getLink: "https://personal-kd-njs.yun.139.com/hcy/file/getDownloadUrl"
  323. },
  324. mount: {
  325. home: ".top_button",
  326. share: ".top-btns"
  327. },
  328. dom: {
  329. enhance: `+<br/>此方式下载不会被 IDM 捕获下载链接`,
  330. normal: `+<br/>此方式下载有可能会被 IDM 捕获下载链接`
  331. }
  332. },
  333. $tcloud: {
  334. api: {
  335. getAccessToken: "https://api.cloud.189.cn/open/oauth2/ssoH5.action",
  336. getLink: "https://api.cloud.189.cn/open/file/getFileDownloadUrl.action"
  337. },
  338. mount: {
  339. home: "[class*=\"FileHead_file-head-left\"]",
  340. share: ".nav-opea"
  341. },
  342. dom: {
  343. enhance: `+<br/>此方式下载不会被 IDM 捕获下载链接`,
  344. normal: `+<br/>此方式下载有可能会被 IDM 捕获下载链接`
  345. }
  346. },
  347. $xunlei: {
  348. api: {
  349. mirror: [
  350. "vod0007-h05-vip-lixian.xunlei.com", "vod0008-h05-vip-lixian.xunlei.com", "vod0009-h05-vip-lixian.xunlei.com", "vod0010-h05-vip-lixian.xunlei.com", "vod0011-h05-vip-lixian.xunlei.com", "vod0012-h05-vip-lixian.xunlei.com", "vod0013-h05-vip-lixian.xunlei.com", "vod0014-h05-vip-lixian.xunlei.com", "vod0067-aliyun08-vip-lixian.xunlei.com", "vod0254-aliyun08-vip-lixian.xunlei.com", "vod0255-aliyun08-vip-lixian.xunlei.com", "vod0256-aliyun08-vip-lixian.xunlei.com", "vod0257-aliyun08-vip-lixian.xunlei.com", "vod0258-aliyun08-vip-lixian.xunlei.com", "vod0259-aliyun08-vip-lixian.xunlei.com", "vod0260-aliyun08-vip-lixian.xunlei.com", "vod0261-aliyun08-vip-lixian.xunlei.com", "vod0262-aliyun08-vip-lixian.xunlei.com", "vod0263-aliyun08-vip-lixian.xunlei.com", "vod0264-aliyun08-vip-lixian.xunlei.com", "vod0265-aliyun08-vip-lixian.xunlei.com", "vod0266-aliyun08-vip-lixian.xunlei.com", "vod0267-aliyun08-vip-lixian.xunlei.com", "vod0554-aliyun06-vip-lixian.xunlei.com", "vod0555-aliyun06-vip-lixian.xunlei.com", "vod0556-aliyun06-vip-lixian.xunlei.com", "vod0680-aliyun08-vip-lixian.xunlei.com", "vod0681-aliyun08-vip-lixian.xunlei.com", "vod0682-aliyun08-vip-lixian.xunlei.com", "vod0683-aliyun08-vip-lixian.xunlei.com", "vod0684-aliyun08-vip-lixian.xunlei.com", "vod0685-aliyun08-vip-lixian.xunlei.com", "vod0686-aliyun08-vip-lixian.xunlei.com", "vod0687-aliyun08-vip-lixian.xunlei.com", "vod0688-aliyun08-vip-lixian.xunlei.com", "vod0689-aliyun08-vip-lixian.xunlei.com", "vod0690-aliyun08-vip-lixian.xunlei.com", "vod0724-aliyun08-vip-lixian.xunlei.com", "vod0725-aliyun08-vip-lixian.xunlei.com", "vod0726-aliyun08-vip-lixian.xunlei.com", "vod0727-aliyun08-vip-lixian.xunlei.com", "vod0728-aliyun08-vip-lixian.xunlei.com", "vod0075.aliyun06.vip.lixian.xunlei.com", "vod0076.aliyun06.vip.lixian.xunlei.com", "vod0077.aliyun06.vip.lixian.xunlei.com", "vod0779-aliyun04-vip-lixian.xunlei.com", "vod0078.aliyun06.vip.lixian.xunlei.com", "vod0780-aliyun04-vip-lixian.xunlei.com", "vod0781-aliyun04-vip-lixian.xunlei.com", "vod0079.aliyun06.vip.lixian.xunlei.com", "vod0080.aliyun06.vip.lixian.xunlei.com", "vod0117.aliyun04.vip.lixian.xunlei.com", "vod0118.aliyun04.vip.lixian.xunlei.com", "vod0119.aliyun04.vip.lixian.xunlei.com", "vod1284-aliyun06-vip-lixian.xunlei.com", "vod1285-aliyun06-vip-lixian.xunlei.com", "vod1363-aliyun06-vip-lixian.xunlei.com", "vod1371-aliyun06-vip-lixian.xunlei.com", "vod1372-aliyun06-vip-lixian.xunlei.com", "vod1426-aliyun06-vip-lixian.xunlei.com", "vod1427-aliyun06-vip-lixian.xunlei.com", "vod1428-aliyun06-vip-lixian.xunlei.com", "vod1429-aliyun06-vip-lixian.xunlei.com", "vod1442-aliyun06-vip-lixian.xunlei.com", "vod1443-aliyun06-vip-lixian.xunlei.com", "vod1444-aliyun06-vip-lixian.xunlei.com", "vod1445-aliyun06-vip-lixian.xunlei.com", "vod1446-aliyun06-vip-lixian.xunlei.com", "vod1447-aliyun06-vip-lixian.xunlei.com", "vod1469-aliyun06-vip-lixian.xunlei.com", "vod1470-aliyun06-vip-lixian.xunlei.com", "vod1471-aliyun06-vip-lixian.xunlei.com", "vod1489-aliyun06-vip-lixian.xunlei.com", "vod1490-aliyun06-vip-lixian.xunlei.com", "vod1491-aliyun06-vip-lixian.xunlei.com", "vod1492-aliyun06-vip-lixian.xunlei.com", "vod1493-aliyun06-vip-lixian.xunlei.com", "vod0215.aliyun06.vip.lixian.xunlei.com", "vod0216.aliyun06.vip.lixian.xunlei.com", "vod0217.aliyun06.vip.lixian.xunlei.com", "vod0218.aliyun06.vip.lixian.xunlei.com", "vod0219.aliyun06.vip.lixian.xunlei.com", "vod0220.aliyun06.vip.lixian.xunlei.com", "vod0241.aliyun08.vip.lixian.xunlei.com", "vod0244.aliyun08.vip.lixian.xunlei.com", "vod0251.aliyun08.vip.lixian.xunlei.com", "vod0252.aliyun08.vip.lixian.xunlei.com", "vod0253.aliyun08.vip.lixian.xunlei.com", "vod0254.aliyun08.vip.lixian.xunlei.com", "vod0255.aliyun08.vip.lixian.xunlei.com", "vod0256.aliyun08.vip.lixian.xunlei.com", "vod0257.aliyun08.vip.lixian.xunlei.com", "vod0260.aliyun08.vip.lixian.xunlei.com", "vod0261.aliyun08.vip.lixian.xunlei.com", "vod0262.aliyun08.vip.lixian.xunlei.com", "vod0263.aliyun08.vip.lixian.xunlei.com", "vod0264.aliyun08.vip.lixian.xunlei.com", "vod0265.aliyun08.vip.lixian.xunlei.com", "vod0266.aliyun08.vip.lixian.xunlei.com", "vod0267.aliyun08.vip.lixian.xunlei.com", "vod3379-aliyun04-vip-lixian.xunlei.com", "vod3380-aliyun04-vip-lixian.xunlei.com", "vod3429-aliyun04-vip-lixian.xunlei.com", "vod3458-aliyun04-vip-lixian.xunlei.com", "vod3459-aliyun04-vip-lixian.xunlei.com", "vod3496-aliyun04-vip-lixian.xunlei.com", "vod3497-aliyun04-vip-lixian.xunlei.com", "vod3498-aliyun04-vip-lixian.xunlei.com", "vod3499-aliyun04-vip-lixian.xunlei.com", "vod3500-aliyun04-vip-lixian.xunlei.com", "vod3501-aliyun04-vip-lixian.xunlei.com", "vod3522-aliyun04-vip-lixian.xunlei.com", "vod3523-aliyun04-vip-lixian.xunlei.com", "vod3533-aliyun04-vip-lixian.xunlei.com", "vod3534-aliyun04-vip-lixian.xunlei.com", "vod3535-aliyun04-vip-lixian.xunlei.com", "vod3536-aliyun04-vip-lixian.xunlei.com", "vod3549-aliyun04-vip-lixian.xunlei.com", "vod3550-aliyun04-vip-lixian.xunlei.com", "vod3551-aliyun04-vip-lixian.xunlei.com", "vod3552-aliyun04-vip-lixian.xunlei.com", "vod3553-aliyun04-vip-lixian.xunlei.com", "vod3554-aliyun04-vip-lixian.xunlei.com", "vod3555-aliyun04-vip-lixian.xunlei.com", "vod0551.aliyun06.vip.lixian.xunlei.com", "vod0552.aliyun06.vip.lixian.xunlei.com", "vod0553.aliyun06.vip.lixian.xunlei.com", "vod0554.aliyun06.vip.lixian.xunlei.com", "vod0555.aliyun06.vip.lixian.xunlei.com", "vod0556.aliyun06.vip.lixian.xunlei.com", "vod0686.aliyun08.vip.lixian.xunlei.com", "vod0687.aliyun08.vip.lixian.xunlei.com", "vod0688.aliyun08.vip.lixian.xunlei.com", "vod0689.aliyun08.vip.lixian.xunlei.com", "vod0724.aliyun08.vip.lixian.xunlei.com", "vod0725.aliyun08.vip.lixian.xunlei.com", "vod0726.aliyun08.vip.lixian.xunlei.com", "vod0727.aliyun08.vip.lixian.xunlei.com", "vod0728.aliyun08.vip.lixian.xunlei.com", "vod0759.aliyun04.vip.lixian.xunlei.com", "vod0760.aliyun04.vip.lixian.xunlei.com", "vod0769.aliyun04.vip.lixian.xunlei.com", "vod0770.aliyun04.vip.lixian.xunlei.com", "vod0771.aliyun04.vip.lixian.xunlei.com", "vod0772.aliyun04.vip.lixian.xunlei.com", "vod0773.aliyun04.vip.lixian.xunlei.com", "vod0774.aliyun04.vip.lixian.xunlei.com", "vod0775.aliyun04.vip.lixian.xunlei.com", "vod0776.aliyun04.vip.lixian.xunlei.com", "vod0777.aliyun04.vip.lixian.xunlei.com", "vod0778.aliyun04.vip.lixian.xunlei.com", "vod0779.aliyun04.vip.lixian.xunlei.com", "vod0780.aliyun04.vip.lixian.xunlei.com", "vod0781.aliyun04.vip.lixian.xunlei.com", "vod3522.aliyun04.vip.lixian.xunlei.com", "vod3523.aliyun04.vip.lixian.xunlei.com", "vod3533.aliyun04.vip.lixian.xunlei.com", "vod3535.aliyun04.vip.lixian.xunlei.com", "vod3550.aliyun04.vip.lixian.xunlei.com", "vod3551.aliyun04.vip.lixian.xunlei.com", "vod3552.aliyun04.vip.lixian.xunlei.com", "vod3553.aliyun04.vip.lixian.xunlei.com", "vod3554.aliyun04.vip.lixian.xunlei.com", "vod3555.aliyun04.vip.lixian.xunlei.com"
  351. ],
  352. getLink: "https://api-pan.xunlei.com/drive/v1/files/"
  353. },
  354. mount: {
  355. home: "[class^=\"FileMenu__menu--\"]",
  356. share: "[class^=\"Share__batchActionBox--\"]"
  357. },
  358. dom: {
  359. enhance: `+<br/>此方式下载不会被 IDM 捕获下载链接,但可以自动命名`,
  360. normal: `+<br/>此方式下载不会被 IDM 捕获下载链接`,
  361. filename: `迅雷云盘下载时可能不会回报文件名称给下载工具,这时需要手动复制文件名称到下载工具中`
  362. }
  363. },
  364. $quark: {
  365. api: {
  366. ua: {
  367. downloadLink: "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) quark-cloud-drive/2.5.20 Chrome/100.0.4896.160 Electron/18.3.5.4-b478491100 Safari/537.36 Channel/pckk_other_ch"
  368. },
  369. getLink: "https://drive.quark.cn/1/clouddrive/file/download?pr=ucpro&fr=pc"
  370. },
  371. mount: {
  372. home: ".btn-operate .btn-main",
  373. share: ".share-btns"
  374. },
  375. dom: {
  376. enhance: `+<br/>此方式下载不会被 IDM 捕获下载链接`,
  377. normal: `+<br/>此方式下载有可能会被 IDM 捕获下载链接`
  378. }
  379. },
  380. $uc: {
  381. api: {
  382. ua: {
  383. downloadLink: "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) uc-cloud-drive/2.5.20 Chrome/100.0.4896.160 Electron/18.3.5.4-b478491100 Safari/537.36 Channel/pckk_other_ch"
  384. },
  385. getLink: "https://pc-api.uc.cn/1/clouddrive/file/download?pr=UCBrowser&fr=pc"
  386. },
  387. mount: {
  388. home: ".btn-operate .btn-main",
  389. share: ".file-info-share-buttom"
  390. },
  391. dom: {
  392. enhance: `+<br/>此方式下载不会被 IDM 捕获下载链接`,
  393. normal: `+<br/>此方式下载有可能会被 IDM 捕获下载链接`
  394. }
  395. },
  396. $123pan: {
  397. api: {
  398. getLink: "https://www.123pan.com/api/file/download_info",
  399. getShareLink: "https://www.123pan.com/api/share/download/info"
  400. },
  401. mount: {
  402. home: "main .homeClass .home-operator .home-operator-button-group",
  403. share: ".conter .rightInfo .qrcode_btn"
  404. },
  405. dom: {
  406. enhance: `+<br/>此方式下载不会被 IDM 捕获下载链接`,
  407. normal: `+<br/>此方式下载有可能会被 IDM 捕获下载链接`
  408. }
  409. }
  410. }
  411. /**
  412. * 基础工具集合
  413. * @author 油小猴
  414. * @author hmjz100
  415. */
  416. let base = {
  417. /**
  418. * 注册 GreaseMonkey-Compatible-Manager 扩展菜单命令
  419. * @author 油小猴
  420. * @author hmjz100
  421. * @description 包含 "设置"、"美化"、"更新" 和 "调试" 四个功能入口
  422. */
  423. registerMenuCommand() {
  424. GM_registerMenuCommand('⚙️ 设置', () => {
  425. base.showSetting();
  426. });
  427. GM_registerMenuCommand('🍃️ 美化', () => {
  428. base.showBeautify();
  429. });
  430. GM_registerMenuCommand('📃 更新', () => {
  431. base.showUpdate();
  432. });
  433. GM_registerMenuCommand('🛠️ 调试', () => {
  434. base.showDebug();
  435. });
  436. },
  437. /**
  438. * 判断 JavaScript 对象类型
  439. * @author 油小猴
  440. * @description 通过 Object.prototype.toString 精确识别对象类型
  441. * @param {*} obj - 待检测对象
  442. * @returns {string} 类型名称(全小写),如:array/number/null/date 等
  443. * @example
  444. * isType([]) // => "array"
  445. * isType(null) // => "null"
  446. */
  447. isType(obj) {
  448. return Object.prototype.toString.call(obj).replace(/^\[object (.+)\]$/, '$1').toLowerCase();
  449. },
  450. /**
  451. * 获取 GreaseMonkey-Compatible-Manager 存储的值
  452. * @author 油小猴
  453. * @param {string} name - 存储键名
  454. * @returns {*} 存储的值
  455. */
  456. getValue(name) {
  457. return GM_getValue(name);
  458. },
  459. /**
  460. * 设置 GreaseMonkey-Compatible-Manager 存储的值
  461. * @author 油小猴
  462. * @param {string|array} path - 存储键名或路径数组
  463. * @param {*} value - 要存储的值
  464. */
  465. setValue(path, value) {
  466. if (base.isType(path) === 'string') {
  467. GM_setValue(path, value);
  468. return;
  469. }
  470. let key = path[0];
  471. let obj = this.getValue(key) || {};
  472. let current = obj;
  473. for (let i = 1; i < path.length - 1; i++) {
  474. let keyPart = path[i];
  475. if (!current[keyPart]) current[keyPart] = "";
  476. current = current[keyPart];
  477. }
  478. current[path[path.length - 1]] = value;
  479. GM_setValue(key, obj);
  480. },
  481. /**
  482. * 删除 GreaseMonkey-Compatible-Manager 存储的值
  483. * @author 油小猴
  484. * @param {string|array} key - 单个键名
  485. */
  486. delValue(key) {
  487. return GM_deleteValue(key);
  488. },
  489. /**
  490. * 从 localStorage 获取存储值
  491. * @description 自动解析 JSON 格式内容
  492. * @author 油小猴
  493. * @param {string} key - 存储键名
  494. * @returns {*} 存储的原始值或解析后的对象
  495. */
  496. getStorage(key) {
  497. try {
  498. return JSON.parse(localStorage.getItem(key));
  499. } catch (e) {
  500. return localStorage.getItem(key);
  501. }
  502. },
  503. /**
  504. * 设置 localStorage 存储值
  505. * @author 油小猴
  506. * @description 自动 `JSON.stringify` `对象` `数组` 类型的数据
  507. * @param {string} key - 存储键名
  508. * @param {*} value - 要存储的值
  509. */
  510. setStorage(key, value) {
  511. if (this.isType(value) === 'object' || this.isType(value) === 'array') {
  512. return localStorage.setItem(key, JSON.stringify(value));
  513. }
  514. return localStorage.setItem(key, value);
  515. },
  516. /**
  517. * 删除 localStorage 存储值
  518. * @author 油小猴
  519. * @description 没什么特别的
  520. * @param {string} key - 存储键名
  521. */
  522. delStorage(key) {
  523. return localStorage.removeItem(key);
  524. },
  525. /**
  526. * 剪贴板写入
  527. * @author 油小猴
  528. * @param {string} text - 要复制的文本内容
  529. */
  530. setClipboard(text) {
  531. GM_setClipboard(text, 'text');
  532. },
  533. /**
  534. * Base64-URI 编码处理
  535. * @author 油小猴
  536. * @author hmjz100
  537. * @description 自动执行 URI 兼容性编码转换
  538. * @param {string} str - 待编码的字符串
  539. * @returns {string} Base64 编码结果字符串
  540. */
  541. encodeBase(str) {
  542. try { str = btoa(str) } catch { }
  543. return str;
  544. },
  545. /**
  546. * Base64-URI 解码处理
  547. * @author 油小猴
  548. * @author hmjz100
  549. * @description 自动执行 URI 兼容性解码转换
  550. * @param {string} str - Base64 编码字符串
  551. * @returns {string} 解码后的原始字符串
  552. */
  553. decodeBase(str) {
  554. try { str = decodeURIComponent(str) } catch { }
  555. try { str = atob(str) } catch { }
  556. try { str = decodeURIComponent(str) } catch { }
  557. return str;
  558. },
  559. /**
  560. * 数字补零格式化
  561. * @author hmjz100
  562. * @description 对 1-9 的数字自动补前导零
  563. * @param {number} i - 待格式化的数字
  564. * @returns {string} 格式化后的字符串(如"05")
  565. */
  566. timeFormat(i) {
  567. if (i >= 0 && i <= 9) {
  568. return "0" + i;
  569. } else {
  570. return i;
  571. }
  572. },
  573. /**
  574. * 获取文件扩展名并转为大写
  575. * @author 油小猴
  576. * @param {string} name - 完整文件名
  577. * @returns {string} 大写的文件扩展名(如 `TXT`)
  578. */
  579. getExtension(name) {
  580. let reg = /(?!\.)\w+$/;
  581. if (reg.test(name)) {
  582. let match = name.match(reg);
  583. return match[0].toUpperCase();
  584. }
  585. return "";
  586. },
  587. /**
  588. * 文件大小格式化
  589. * @author hmjz100
  590. * @description 自动转换单位到最合适的存储单位(如 `1.2MB`)
  591. * @param {number} value - 文件字节大小
  592. * @returns {string} 可读格式的大小描述
  593. */
  594. sizeFormat(value = 0) {
  595. try { value = Number(value) } catch { }
  596. if (value === +value) {
  597. let unit = ["字节(B)", "千字节(KB)", "兆字节(MB)", "吉字节(GB)", "太字节(TB)", "拍字节(PB)", "艾字节(EB)", "泽字节(ZB)", "尧字节(YB)"];
  598. if (value === 0) {
  599. return "0字节(B)";
  600. } else {
  601. let index = Math.floor(Math.log(value) / Math.log(1024));
  602. let size = value / Math.pow(1024, index);
  603. size = size.toFixed(2);
  604. return size + unit[index];
  605. }
  606. }
  607. return "";
  608. },
  609. /**
  610. * 将剩余时间(秒)格式化为可读的时间字符串
  611. *
  612. * @param {number} remainingTimeSeconds 剩余总秒数(支持小数)
  613. * @returns {string} 格式化后的时间字符串,包含以下可能格式:
  614. * - "X天 HH时:MM分:SS秒"(超过1天)
  615. * - "HH时:MM分:SS秒"(超过1小时)
  616. * - "MM分:SS秒"(超过1分钟)
  617. * - "SS秒"(1分钟内)
  618. * - "即将完成"(0秒时)
  619. * - "计算中..."(无效输入时)
  620. *
  621. * @example
  622. * formatRemainingTime(86400) // "1天 00时:00分:00秒"
  623. * formatRemainingTime(3661.5) // "01时:01分:01秒"
  624. * formatRemainingTime(0) // "即将完成"
  625. * formatRemainingTime(-5) // "计算中..."
  626. * formatRemainingTime(NaN) // "计算中..."
  627. */
  628. rtimeFormat(remainingTimeSeconds) {
  629. if (!Number.isFinite(remainingTimeSeconds) || remainingTimeSeconds < 0) {
  630. return '计算中...';
  631. }
  632. let remainingDays = Math.floor(remainingTimeSeconds / (60 * 60 * 24));
  633. remainingTimeSeconds %= (60 * 60 * 24);
  634. let remainingHours = Math.floor(remainingTimeSeconds / (60 * 60));
  635. remainingTimeSeconds %= (60 * 60);
  636. let remainingMinutes = Math.floor(remainingTimeSeconds / 60);
  637. let remainingSeconds = Math.floor(remainingTimeSeconds % 60);
  638. if (remainingDays > 0) {
  639. return `${remainingDays}天 ${base.timeFormat(remainingHours)}时:${base.timeFormat(remainingMinutes)}分:${base.timeFormat(remainingSeconds)}秒`;
  640. } else if (remainingHours > 0) {
  641. return `${base.timeFormat(remainingHours)}时:${base.timeFormat(remainingMinutes)}分:${base.timeFormat(remainingSeconds)}秒`;
  642. } else if (remainingMinutes > 0) {
  643. return `${base.timeFormat(remainingMinutes)}分:${base.timeFormat(remainingSeconds)}秒`;
  644. } else if (remainingSeconds > 0) {
  645. return `${remainingSeconds}秒`;
  646. } else {
  647. return '即将完成';
  648. }
  649. },
  650. /**
  651. * 文件列表排序
  652. * @author 油小猴
  653. * @description 按中文拼音顺序对文件数组进行排序
  654. * @param {Array} arr - 包含文件对象的数组
  655. * @param {string} arr[].filename - 文件名属性(兼容 server_filename)
  656. */
  657. sortByName(arr) {
  658. let handle = () => {
  659. return (a, b) => {
  660. let p1 = a.filename ? a.filename : a.server_filename;
  661. let p2 = b.filename ? b.filename : b.server_filename;
  662. return p1.localeCompare(p2, "zh-CN");
  663. };
  664. };
  665. arr.sort(handle());
  666. },
  667. /**
  668. * 文件名安全处理
  669. * @author 油小猴
  670. * @description 替换非法字符为下划线
  671. * @param {string} name - 原始文件名
  672. * @returns {string} 修正后的安全文件名
  673. */
  674. fixFilename(name) {
  675. let replace = /[!?&|`"'*\/:<>\\]/g
  676. return name.replace(replace, '_');
  677. },
  678. /**
  679. * 生成 cURL 下载命令
  680. * @author 油小猴
  681. * @author hmjz100
  682. * @description 根据终端类型生成对应 curl 命令,支持断点续传,自动处理文件名特殊字符
  683. * @param {string} link - 下载链接
  684. * @param {string} filename - 文件名
  685. * @param {string} [headers] - 自定义请求头参数(可选)
  686. * @returns {string} 编码后的 curl 命令字符串
  687. */
  688. convertLinkToCurl(link, filename, headers) {
  689. let terminal = base.getValue('setting_curl_terminal');
  690. filename = base.fixFilename(filename);
  691. return `${terminal !== 'wp' ? 'curl' : 'curl.exe'} -L -C - "${link}" -o "${filename}"${headers ? (" " + headers) : ""}`;
  692. },
  693. /**
  694. * 生成 Aria2 下载命令
  695. * @author 油小猴
  696. * @author hmjz100
  697. * @description 将链接转换为 Aria2 格式命令,自动处理文件名特殊字符
  698. * @param {string} link - 下载链接
  699. * @param {string} filename - 文件名
  700. * @param {string} [headers] - 自定义请求头参数(可选)
  701. * @returns {string} 编码后的 aria2c 命令字符串
  702. */
  703. convertLinkToAria2(link, filename, headers) {
  704. filename = base.fixFilename(filename);
  705. return `aria2c "${link}" --out "${filename}"${headers ? (" " + headers) : ""}`;
  706. },
  707. /**
  708. * 生成 BC 协议下载链接
  709. * @author 油小猴
  710. * @author hmjz100
  711. * @description 将链接转换为 BC 协议格式,自动处理 URL 编码
  712. * @param {string} link - 下载链接
  713. * @param {string} filename - 文件名
  714. * @param {string} [headers] - 自定义请求头参数(可选)
  715. * @returns {string} 编码后的 BC 协议 URL
  716. */
  717. convertLinkToBitComet(link, filename, headers) {
  718. filename = base.fixFilename(filename);
  719. let bc = `AA/${encodeURIComponent(filename)}/?url=${encodeURIComponent(link)}${headers ? ("&" + headers) : ""}ZZ`;
  720. return `bc://http/${base.encodeBase(bc)}`;
  721. },
  722. /**
  723. * 发送链接到 Aria2 下载器
  724. * @author 油小猴
  725. * @author hmjz100
  726. * @description Aria2 下载必备
  727. * @param {string} link - 下载链接
  728. * @param {string} filename - 文件名
  729. * @param {Array} [headers] - 自定义请求头参数(可选)
  730. * @returns {Promise<'success'|'fail'>} 发送态结果
  731. */
  732. async sendLinkToAria2(link, filename, headers) {
  733. let list = base.getValue('setting_aria2_rpc');
  734. let selected = list.find(i => i.default);
  735. let rpc = {
  736. domain: selected.domain,
  737. port: selected.port,
  738. path: selected.path,
  739. dir: selected.dir,
  740. token: selected.token
  741. };
  742. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  743. let dir = (rpc.dir !== null && rpc.dir !== "") ? rpc.dir : undefined;
  744. let data = {
  745. id: new Date().getTime(),
  746. jsonrpc: '2.0',
  747. method: 'aria2.addUri',
  748. params: [`token:${rpc.token}`, [link], {
  749. dir,
  750. out: filename,
  751. header: headers
  752. }]
  753. };
  754. try {
  755. let res = await base.post(url, data, {}, "");
  756. if (res.result) return 'success';
  757. return 'fail';
  758. } catch (e) {
  759. return 'fail';
  760. }
  761. },
  762. /**
  763. * 发送链接到比特彗星下载器
  764. * @author hmjz100
  765. * @description 比特彗星下载必备
  766. * @param {string} link - 下载链接
  767. * @param {string} filename - 文件名
  768. * @param {Array} [headers] - 自定义请求头参数(可选)
  769. * @returns {Promise<'success'|'fail'>} 发送态结果
  770. */
  771. async sendLinkToBitcomet(link, filename, headers) {
  772. let list = base.getValue('setting_bitcomet_rpc');
  773. let selected = list.find(i => i.default);
  774. let rpc = {
  775. domain: selected.domain,
  776. port: selected.port,
  777. path: selected.path,
  778. dir: selected.dir,
  779. authName: selected.authName,
  780. authPass: selected.authPass,
  781. };
  782. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  783. let data = new URLSearchParams();
  784. data.append('url', link);
  785. if (rpc.dir !== null && rpc.dir !== "") data.append('save_path', rpc.dir);
  786. data.append('file_name', filename);
  787. data.append('connection', 200);
  788. if (headers && base.isType(headers) === 'object') {
  789. for (const [key, value] of Object.entries(headers)) {
  790. data.append(key, value);
  791. }
  792. }
  793. try {
  794. let res = await base.post(url, data, {
  795. "Authorization": `Basic ${base.encodeBase(rpc.authName + ":" + rpc.authPass)}`,
  796. "Content-Type": "application/x-www-form-urlencoded"
  797. }, "blob");
  798. if (res.response && res?.responseText?.includes('Add task failed!')) {
  799. return 'fail';
  800. } else {
  801. return "success";
  802. }
  803. } catch (e) {
  804. return "success";
  805. }
  806. },
  807. /**
  808. * 发送链接到 AB Download Manager 下载器
  809. * @author hmjz100
  810. * @description AB Download Manager 下载必备
  811. * @param {string} link - 下载链接
  812. * @param {string} filename - 文件名
  813. * @param {Array} [headers] - 自定义请求头参数(可选)
  814. * @returns {Promise<'success'|'fail'>} 发送态结果
  815. */
  816. async sendLinkToABDM(link, filename, headers) {
  817. let newHeaders = {};
  818. for (let key in headers) {
  819. newHeaders[key.toLowerCase().split('-').map(word => word.charAt(0).toUpperCase() + word.slice(1)).join('-')] = headers[key];
  820. }
  821. headers = { "User-Agent": navigator.userAgent, "Origin": location.origin, "Referer": `${location.origin}/`, "DNT": "1", ...newHeaders };
  822. let list = base.getValue('setting_abdm_rpc');
  823. let selected = list.find(i => i.default);
  824. let rpc = {
  825. domain: selected.domain,
  826. port: selected.port,
  827. dir: selected.dir
  828. };
  829. let url = `${rpc.domain}:${rpc.port}/start-headless-download`;
  830. let data = {
  831. "downloadSource": {
  832. "name": filename,
  833. "description": "LinkSwift",
  834. "link": link,
  835. "headers": headers,
  836. "downloadPage": headers["Referer"]
  837. },
  838. "name": filename
  839. }
  840. if (rpc.dir) data.folder = rpc.dir;
  841. try {
  842. let res = await base.post(url, data, { "Content-Type": "text/plain;charset=UTF-8" }, "text");
  843. if (res === "OK") return "success";
  844. return 'fail';
  845. } catch (e) {
  846. return 'fail';
  847. }
  848. },
  849. /**
  850. * Blob 文件下载
  851. * @author 油小猴
  852. * @description 通过创建临时链接实现文件下载
  853. * @param {Blob} blob - 要下载的 Blob 对象
  854. * @param {string} filename - 下载时提示保存的文件名
  855. */
  856. blobDownload(blob, filename) {
  857. if (blob instanceof Blob) {
  858. let url = URL.createObjectURL(blob);
  859. let a = document.createElement('a');
  860. a.href = url;
  861. a.download = filename;
  862. a.click();
  863. URL.revokeObjectURL(url);
  864. }
  865. },
  866. /**
  867. * 可跨域 xmlhttpRequest 请求
  868. * @author hmjz100
  869. * @description 封装 `GreaseMonkey-Compatible_xmlhttpRequest` 实现的跨域请求,与原始函数参数相同
  870. * @param {Object} option - 请求配置对象
  871. * @returns {XMLHttpRequest} 请求对象实例
  872. */
  873. xmlHttpRequest(option) {
  874. let request = (base.isType(GM_xmlhttpRequest) !== "undefined") ? GM_xmlhttpRequest : GM.xmlHttpRequest;
  875. if (request && base.isType(request) === 'function') return request(option);
  876. },
  877. /**
  878. * 发送 POST 请求
  879. * @author 油小猴
  880. * @author hmjz100
  881. * @description 一般用于请求 API,支持自动重试和数据格式化,内置错误处理,请求数据自动 `JSON.stringify`
  882. * @param {string} url - 请求地址
  883. * @param {Object|string} data - 请求数据
  884. * @param {Object} headers - 请求头配置
  885. * @param {string} [type='json'] - 响应类型(支持 `json`, `blob` 等)
  886. * @returns {Promise} 包含响应数据的 `Promise` 对象
  887. */
  888. post(url, data, headers, type = "json") {
  889. let exampleData = data;
  890. if (this.isType(data) === 'object' || this.isType(data) === 'array') {
  891. data = JSON.stringify(data);
  892. } else if (this.isType(data) === 'urlsearchparams') {
  893. exampleData = Object.fromEntries(data);
  894. }
  895. let newHeaders = {};
  896. for (let key in headers) {
  897. newHeaders[key.toLowerCase().split('-').map(word => word.charAt(0).toUpperCase() + word.slice(1)).join('-')] = headers[key];
  898. }
  899. headers = { "Content-Type": "application/json;charset=utf-8", "User-Agent": navigator.userAgent, "Origin": location.origin, "Referer": `${location.origin}/`, "Dnt": "", ...newHeaders };
  900. return new Promise((resolve, reject) => {
  901. let sendRequest = () => {
  902. base.xmlHttpRequest({
  903. method: "POST", url, headers, data,
  904. responseType: type,
  905. onloadstart() {
  906. console.log('【LinkSwift】Post(start)\n请求地址:' + url + '\n请求头部:', headers, '\n请求数据:', exampleData);
  907. },
  908. onload: function (res) {
  909. console.log('【LinkSwift】Post(load)\n请求地址:' + url, '\n请求结果:', res);
  910. try { res.decodeResponse = JSON.parse(res.response) } catch { }
  911. type === 'blob' ? resolve(res) : resolve(res.decodeResponse || res.response || res.responseText);
  912. },
  913. onerror: async function (err) {
  914. console.error('【LinkSwift】Post(error)\n请求出现错误,可能是网络问题。', err);
  915. reject(err);
  916. },
  917. });
  918. };
  919. sendRequest(); // 初始请求
  920. });
  921. },
  922. /**
  923. * 发送 GET 请求
  924. * @author 油小猴
  925. * @author hmjz100
  926. * @description 一般用于下载文件,支持进度监控、文件下载和自动重试,可处理被下载工具捕获特殊逻辑
  927. * @param {string} url - 请求地址
  928. * @param {Object} headers - 请求头配置
  929. * @param {string} [type='json'] - 响应类型
  930. * @param {Object} [extra] - 附加参数(需包含 `filename` 和 `index` 属性)
  931. * @returns {Promise} 包含响应数据的 `Promise` 对象
  932. */
  933. get(url, headers, type, extra) {
  934. let newHeaders = {};
  935. for (let key in headers) {
  936. newHeaders[key.toLowerCase().split('-').map(word => word.charAt(0).toUpperCase() + word.slice(1)).join('-')] = headers[key];
  937. }
  938. headers = { "User-Agent": navigator.userAgent, "Origin": location.origin, "Referer": `${location.origin}/`, "Dnt": "", ...newHeaders };
  939. return new Promise((resolve, reject) => {
  940. let sendRequest = () => {
  941. let requestObj = base.xmlHttpRequest({
  942. method: "GET", url, headers,
  943. responseType: type || 'json',
  944. onload: function (res) {
  945. try { res.decodeResponse = JSON.parse(res.response) } catch { }
  946. if (res.status === 204) {
  947. console.log('【LinkSwift】Get(load)\n\x1B[31m该请求已被某个下载工具捕获。' + (res.statusText ? ("\n\x1B[0m工具提示:\x1B[31m" + res.statusText) : "") + '\x1B[0m\n请求地址:' + url + '\n请求头部:', headers, '\n请求结果:', res);
  948. requestObj.abort();
  949. temp.idm[extra.index] = true;
  950. return;
  951. }
  952. if (type === 'blob') {
  953. console.log('【LinkSwift】Get(load) Blob\n请求地址:' + url, '\n请求结果:', res);
  954. res.status === 200 && base.blobDownload(res.response, extra.filename);
  955. resolve(res);
  956. } else {
  957. console.log('【LinkSwift】Get(load)\n请求地址:' + url + '\n请求头部:', headers, '\n请求结果:', res);
  958. resolve(res.decodeResponse || res.response || res.responseText);
  959. }
  960. },
  961. onprogress: function (res) {
  962. if (res.status === 204) {
  963. console.log('【LinkSwift】Get(progress)\n\x1B[31m该请求已被某个下载工具捕获。' + (res.statusText ? ("\n\x1B[0m工具提示:\x1B[31m" + res.statusText) : "") + '\x1B[0m\n请求地址:' + url + '\n请求头部:', headers, '\n请求结果:', res);
  964. requestObj.abort();
  965. temp.idm[extra.index] = true;
  966. return;
  967. }
  968. if (extra && res.loaded && res.total && extra.filename && extra.index) {
  969. res.total > 0 ? temp.progress[extra.index] = (res.loaded * 100 / res.total) : temp.progress[extra.index] = 0.00;
  970. }
  971. },
  972. onloadstart(res) {
  973. if (res.status === 204) {
  974. console.log('【LinkSwift】Get(start)\n\x1B[31m该请求已被某个下载工具捕获。' + (res.statusText ? ("\n\x1B[0m工具提示:\x1B[31m" + res.statusText) : "") + '\x1B[0m\n请求地址:' + url + '\n请求头部:', headers, '\n请求结果:', res);
  975. requestObj.abort();
  976. temp.idm[extra.index] = true;
  977. return;
  978. }
  979. console.log('【LinkSwift】Get(start)\n请求地址:' + url + '\n请求头部:', headers);
  980. if (extra && extra.filename && extra.index) temp.request[extra.index] = requestObj;
  981. },
  982. onerror: async function (err) {
  983. console.error('【LinkSwift】Get(error)\n请求出现错误,可能是网络问题。', err);
  984. reject(err);
  985. },
  986. });
  987. };
  988. sendRequest(); // 初始请求
  989. });
  990. },
  991. /**
  992. * 获取最终重定向 URL
  993. * @author 油小猴
  994. * @author hmjz100
  995. * @description 手动追踪 HTTP 30x 重定向,返回最终访问地址
  996. * @param {string} url - 初始请求地址
  997. * @param {Object} headers - 请求头配置
  998. * @param {boolean} unLimit - 是否完整请求整个文件而不是仅请求请求头
  999. * @returns {Promise<string>} 最终 URL 地址
  1000. */
  1001. getFinalUrl(url, headers, unLimit) {
  1002. return new Promise((resolve, reject) => {
  1003. if (unLimit === true) {
  1004. base.xmlHttpRequest({
  1005. method: "GET",
  1006. url: url,
  1007. headers: headers,
  1008. onloadstart: () => {
  1009. console.log('【LinkSwift】Get(start) FinalUrl\n请求地址:' + url + '\n请求头部:', headers);
  1010. },
  1011. onload: function (res) {
  1012. console.log('【LinkSwift】Get(load) FinalUrl\n请求地址:' + res.finalUrl + '\n响应状态:' + res.status);
  1013. let status = res.status;
  1014. if (status >= 300 && status < 400) {
  1015. base.getFinalUrl(res.finalUrl, headers, maxRetries, currentRetry + 1).then(resolve).catch(reject);
  1016. } else {
  1017. resolve(res.finalUrl);
  1018. }
  1019. },
  1020. onerror: async function (err) {
  1021. console.error('【LinkSwift】Get(error) FinalUrl\n请求出现错误,可能是网络问题。', err);
  1022. reject(err);
  1023. }
  1024. });
  1025. } else {
  1026. base.xmlHttpRequest({
  1027. method: "HEAD",
  1028. url: url,
  1029. headers: headers,
  1030. onloadstart: () => {
  1031. console.log('【LinkSwift】Head(start) FinalUrl\n请求地址:' + url + '\n请求头部:', headers);
  1032. },
  1033. onload: function (res) {
  1034. console.log('【LinkSwift】Head(load) FinalUrl\n请求地址:' + res.finalUrl + '\n响应状态:' + res.status);
  1035. let status = res.status;
  1036. if (status >= 300 && status < 400) {
  1037. base.getFinalUrl(res.finalUrl, headers, maxRetries, currentRetry + 1).then(resolve).catch(reject);
  1038. } else {
  1039. resolve(res.finalUrl);
  1040. }
  1041. },
  1042. onerror: async function (err) {
  1043. console.error('【LinkSwift】Head(error) FinalUrl\n请求出现错误,可能是网络问题。', err);
  1044. reject(err);
  1045. }
  1046. });
  1047. }
  1048. });
  1049. },
  1050. /**
  1051. * Aria2 RPC 服务测试
  1052. * @author hmjz100
  1053. * @description 验证 `JSON-RPC` 接口可用性
  1054. * @param {string} domain - 服务域名
  1055. * @param {string} port - 服务端口
  1056. * @param {string} path - RPC 路径
  1057. * @param {string} token - 认证令牌
  1058. * @returns {Promise<'success'|'fail'>} 连接状态结果
  1059. */
  1060. async testConnectToAria2(domain, port, path, token) {
  1061. return new Promise((resolve, reject) => {
  1062. let rpc = { domain, port, path, token };
  1063. let url = `${rpc.domain}:${rpc.port}${rpc.path}`;
  1064. let rpcData = {
  1065. id: new Date().getTime(),
  1066. jsonrpc: '2.0',
  1067. method: 'aria2.getVersion',
  1068. params: [`token:${rpc.token}`]
  1069. };
  1070. base.xmlHttpRequest({
  1071. method: "POST", url, headers: {}, data: JSON.stringify(rpcData),
  1072. responseType: 'json',
  1073. onloadstart() {
  1074. console.log('【LinkSwift】Post(start) Aria2Test\n请求地址:' + url + '\n请求内容:', rpcData);
  1075. },
  1076. onload: function (res) {
  1077. console.log('【LinkSwift】Post(load) Aria2Test\n请求地址:' + url + '\n请求结果:', res);
  1078. if (!res.response) return resolve("fail");
  1079. if (res.response?.error) {
  1080. resolve("fail");
  1081. } else {
  1082. resolve("success");
  1083. }
  1084. },
  1085. onerror: function (err) {
  1086. console.error('【LinkSwift】Post(error) Aria2Test\n请求失败', err);
  1087. resolve("fail");
  1088. },
  1089. });
  1090. });
  1091. },
  1092. /**
  1093. * AB Download Manager RPC 服务测试
  1094. * @author hmjz100
  1095. * @description 验证 `JSON-RPC` 接口可用性
  1096. * @param {string} domain - 服务域名
  1097. * @param {string} port - 服务端口
  1098. * @returns {Promise<'success'|'fail'>} 连接状态结果
  1099. */
  1100. async testConnectToABDM(domain, port) {
  1101. return new Promise((resolve, reject) => {
  1102. let rpc = { domain, port };
  1103. let url = `${rpc.domain}:${rpc.port}/ping`;
  1104. base.xmlHttpRequest({
  1105. method: "POST", url, headers: {}, data: new Date().getTime(),
  1106. responseType: 'text',
  1107. onloadstart() {
  1108. console.log('【LinkSwift】Post(start) ABDMTest\n请求地址:' + url + '\n请求内容:', new Date().getTime());
  1109. },
  1110. onload: function (res) {
  1111. console.log('【LinkSwift】Post(load) ABDMTest\n请求地址:' + url + '\n请求结果:', res);
  1112. if (!res.response || res.response !== "pong") return resolve("fail");
  1113. resolve("success");
  1114. },
  1115. onerror: function (err) {
  1116. console.error('【LinkSwift】Post(error) ABDMTest\n请求失败', err);
  1117. resolve("fail");
  1118. },
  1119. });
  1120. });
  1121. },
  1122. /**
  1123. * 重置请求相关数据
  1124. * @author 油小猴
  1125. * @description 中止所有进行中的请求,清除进度记录和定时器
  1126. */
  1127. _resetAllData() {
  1128. temp.progress = {};
  1129. $.each(temp.request, function (key) {
  1130. (temp.request[key]).abort();
  1131. });
  1132. $.each(temp.ins, function (key) {
  1133. clearInterval(temp.ins[key]);
  1134. });
  1135. temp.idm = {};
  1136. temp.ins = {};
  1137. temp.request = {};
  1138. },
  1139. /**
  1140. * 重置请求相关数据
  1141. * @author 油小猴
  1142. * @description 中止指定的进行中的请求,清除进度记录和定时器
  1143. */
  1144. _resetData(i) {
  1145. temp.ins[i] && clearInterval(temp.ins[i]);
  1146. temp.request[i] && temp.request[i].abort();
  1147. temp.progress[i] = 0;
  1148. temp.idm[i] = false;
  1149. },
  1150. /**
  1151. * 将对象转换为 URL 编码字符串
  1152. * @author 油小猴
  1153. * @description 递归处理嵌套数组,自动进行 URI 编码
  1154. * @param {Object} obj - 待转换的键值对对象
  1155. * @returns {string} URL 编码格式字符串(如`key1=value1&key2=value2`)
  1156. */
  1157. stringify(obj) {
  1158. let str = "";
  1159. for (let key in obj) {
  1160. if (obj.hasOwnProperty(key)) {
  1161. let value = obj[key];
  1162. if (Array.isArray(value)) {
  1163. for (let i = 0; i < value.length; i++) {
  1164. str += encodeURIComponent(key) + '=' + encodeURIComponent(value[i]) + '&';
  1165. }
  1166. } else {
  1167. str += encodeURIComponent(key) + '=' + encodeURIComponent(value) + '&';
  1168. }
  1169. }
  1170. }
  1171. return str.slice(0, -1); // 去掉末尾的 "&"
  1172. },
  1173. /**
  1174. * 动态注入样式表
  1175. * @author 油小猴
  1176. * @author hmjz100
  1177. * @description 支持 `样式标签` `外链CSS` 注入,提供精准的 DOM 定位和插入位置控制
  1178. * @param {string} id - 样式元素 ID
  1179. * @param {'style'|'link'} tag - 标签类型(`style` 或 `link`)
  1180. * @param {string} css - CSS 内容或外链 URL
  1181. * @param {string} [element='.{mount}'] - 定位基准元素选择器
  1182. * @param {'before'|'after'|'prepend'|'append'} [position='append'] - 插入位置
  1183. */
  1184. addStyle(id, tag = 'style', css, element = `.${mount}`, position = "append") {
  1185. base.waitForKeyElements(element, (element) => {
  1186. let $styleDom = $(`#${id}`);
  1187. let $style = $(`<${tag}>`, {
  1188. rel: 'stylesheet',
  1189. id: id
  1190. });
  1191. tag === 'style' ? $style.html(css.trim().replace(/\t/g, "").replace(/\r\n|\n\r|\n|\r/g, '\n').replace(/\n+/g, '\n')) : $style.attr('href', css);
  1192. if ($styleDom.length) {
  1193. $styleDom.replaceWith($style);
  1194. return true;
  1195. }
  1196. if (position === "before") {
  1197. element.before($style);
  1198. } else if (position === "after") {
  1199. element.after($style);
  1200. } else if (position === "prepend") {
  1201. element.prepend($style);
  1202. } else {
  1203. element.append($style);
  1204. }
  1205. return true;
  1206. }, true);
  1207. },
  1208. /**
  1209. * 十六进制颜色转 RGBA
  1210. * @author hmjz100
  1211. * @description 支持 4 位和 8 位十六进制格式,自动解析透明度通道
  1212. * @param {string} hex - 十六进制颜色值(如 `#09f` 或 `#0099ffaa` )
  1213. * @returns {string} RGBA 格式字符串(如 `15, 170, 255, 0.67`)
  1214. */
  1215. hexToRgba(hex) {
  1216. // 去掉 # 号
  1217. hex = hex.replace(/^#/, "");
  1218. // 如果是四位十六进制颜色值,转换为八位
  1219. if (hex.length === 4) {
  1220. hex = hex[0] + hex[0] + hex[1] + hex[1] + hex[2] + hex[2] + hex[3] + hex[3];
  1221. }
  1222. // 解析 RGB 分量
  1223. let r = parseInt(hex.substring(0, 2), 16);
  1224. let g = parseInt(hex.substring(2, 4), 16);
  1225. let b = parseInt(hex.substring(4, 6), 16);
  1226. let a = "";
  1227. // 如果是八位十六进制颜色值,解析 alpha 通道
  1228. if (hex.length === 8) {
  1229. a = parseInt(hex.substring(6, 8), 16) / 255; // 将 alpha 值转换为 0 到 1 之间的小数
  1230. a = ',' + a
  1231. }
  1232. // 返回 rgba 格式字符串
  1233. return r + ', ' + g + ', ' + b + a;
  1234. },
  1235. /**
  1236. * RGBA 颜色转十六进制
  1237. * @author hmjz100
  1238. * @description 支持透明度转换,自动补全缩写格式
  1239. * @param {string} rgba - RGBA 格式颜色值(如 `rgba(15,170,255,0.67)`)
  1240. * @returns {string} 十六进制颜色值(如 `#09aaffaa`)
  1241. */
  1242. rgbaToHex(rgba) {
  1243. // 去掉前缀 "rgba" 或 "rgb" 并移除空格
  1244. rgba = rgba.replace(/^(rgba?|RGBA?)\(/, "").replace(/\s+/g, "").replace(')', "");
  1245. // 将颜色值分割为数组
  1246. let [r, g, b, a] = rgba.split(',');
  1247. // 将 RGB 转换为十六进制
  1248. r = parseInt(r).toString(16).padStart(2, '0');
  1249. g = parseInt(g).toString(16).padStart(2, '0');
  1250. b = parseInt(b).toString(16).padStart(2, '0');
  1251. // 如果存在 alpha 通道,处理透明度值
  1252. if (a !== undefined) {
  1253. // 将 alpha 转换为 0 到 255 的十六进制
  1254. a = Math.round(parseFloat(a) * 255).toString(16).padStart(2, '0');
  1255. return `#${r}${g}${b}${a}`;
  1256. }
  1257. // 如果没有 alpha 通道,返回标准六位的十六进制颜色
  1258. return `#${r}${g}${b}`;
  1259. },
  1260. /**
  1261. * 自适应 CSS 颜色替换
  1262. * @author hmjz100
  1263. * @description 支持全局样式替换和资源路径修正,处理颜色渐变过渡效果
  1264. * @param {string} cssText - 原始 CSS 内容
  1265. * @param {string} baseURI - 资源基础路径
  1266. * @param {'default'|'other'} type - 替换模式(默认模式包含过渡效果)
  1267. * @param {Array<[string, string]>} colorMap - 颜色映射表(旧颜色 → 新颜色)
  1268. * @returns {string} 处理后的 CSS 内容
  1269. */
  1270. adaptiveStyleOverride(cssText, baseURI, type, colorMap) {
  1271. if (!cssText) return "";
  1272. if (baseURI) {
  1273. // 替换相对路径资源为绝对路径
  1274. cssText = cssText.replace(/url\s*\(\s*(['"]?)(.*?)\1\s*\)/g, (match, quote, url) => {
  1275. if (url && !/^(data:|https?:|\/\/)/i.test(url)) {
  1276. try {
  1277. let absoluteURL = new URL(url, baseURI).href;
  1278. return `url(${absoluteURL})`;
  1279. } catch (e) {
  1280. return match;
  1281. }
  1282. }
  1283. return match;
  1284. });
  1285. }
  1286. // 处理默认颜色列表
  1287. config.base.dom.theme.forEach(item => {
  1288. let oldColor = item.color;
  1289. cssText = cssText.replace(new RegExp(base.hexToRgba(oldColor), 'ig'), base.hexToRgba(temp.color));
  1290. cssText = cssText.replace(new RegExp(oldColor, 'ig'), temp.color);
  1291. });
  1292. // 处理 colorMap
  1293. if (type === 'other') {
  1294. colorMap.forEach(function (colorPair) {
  1295. let oldColor = colorPair[0];
  1296. let newColor = colorPair[1];
  1297. // 生成旧颜色的三种形式:原样、全大写、全小写
  1298. const variants = [
  1299. oldColor,
  1300. oldColor.toUpperCase(),
  1301. oldColor.toLowerCase()
  1302. ];
  1303. // 使用 Set 去重
  1304. const uniqueVariants = [...new Set(variants)];
  1305. uniqueVariants.forEach(variant => {
  1306. const regex = new RegExp(variant, 'g');
  1307. cssText = cssText.replace(regex, newColor);
  1308. });
  1309. });
  1310. return cssText;
  1311. }
  1312. if (colorMap) {
  1313. colorMap.forEach(function (colorPair) {
  1314. let oldColor = colorPair[0];
  1315. let newColor = colorPair[1];
  1316. // 生成三种形式
  1317. const variants = [
  1318. oldColor,
  1319. oldColor.toUpperCase(),
  1320. oldColor.toLowerCase()
  1321. ];
  1322. const uniqueVariants = [...new Set(variants)];
  1323. if (oldColor.includes("#")) {
  1324. // 替换带属性块的情况(添加 transition)
  1325. uniqueVariants.forEach(variant => {
  1326. const regexWithBlock = new RegExp(variant + '(.*?)}', 'gi');
  1327. cssText = cssText.replace(regexWithBlock, newColor + '$1; transition:all.2s}');
  1328. });
  1329. // 最后再统一替换剩下的
  1330. uniqueVariants.forEach(variant => {
  1331. cssText = cssText.replace(new RegExp(variant, 'gi'), newColor);
  1332. });
  1333. } else {
  1334. // 普通字符串替换
  1335. uniqueVariants.forEach(variant => {
  1336. cssText = cssText.replace(new RegExp(variant, 'gi'), newColor);
  1337. });
  1338. }
  1339. });
  1340. }
  1341. return cssText;
  1342. },
  1343. /**
  1344. * 自适应全局主题颜色修改器
  1345. * @author hmjz100
  1346. * @description 自动遍历并替换 `页面所有样式表` `SVG 元素` 的颜色值
  1347. * @param {Array<[string, string]>} colorMap - 颜色映射表
  1348. * @param {'default'|'other'} type - 替换模式
  1349. */
  1350. adaptiveThemeOverride(colorMap, type) {
  1351. base.waitForKeyElements(`[id^="${mount}-ColorUI-"]`, function (tag) {
  1352. if (tag.html() === base.adaptiveStyleOverride(tag.text(), "", type, colorMap)) return;
  1353. let cssText = base.adaptiveStyleOverride(tag.text(), "", type, colorMap);
  1354. base.addStyle(tag.attr("id"), 'style', cssText, tag[0]);
  1355. return true;
  1356. }, true)
  1357. base.waitForKeyElements(`[data-pl-colored]`, function (tag) {
  1358. if (tag.attr("data-pl-colored") === temp.color) return;
  1359. let originalStyle = tag.attr("style");
  1360. if (!originalStyle) return;
  1361. let newStyle = base.adaptiveStyleOverride(originalStyle, "", type, colorMap);
  1362. if (newStyle !== originalStyle) {
  1363. tag.attr("style", newStyle);
  1364. }
  1365. return true;
  1366. }, true);
  1367. let count = 0;
  1368. if (!temp.colored) {
  1369. base.waitForKeyElements('link[rel="stylesheet"]', function (tag) {
  1370. if (!tag.parent().length || !tag.attr('href')) return;
  1371. let href = tag.attr('href');
  1372. try {
  1373. href = new URL(href, location.href).href;
  1374. } catch (e) {
  1375. return;
  1376. }
  1377. fetch(href)
  1378. .then(response => response.text())
  1379. .then(responseText => {
  1380. let id = `${mount}-ColorUI-` + href.replace(/[^\w]/g, "_");
  1381. let cssText = base.adaptiveStyleOverride(responseText, href, type, colorMap);
  1382. if (responseText === base.adaptiveStyleOverride(responseText, href, type, colorMap)) return;
  1383. base.addStyle(id, 'style', cssText, tag[0], "after");
  1384. })
  1385. }, true);
  1386. base.waitForKeyElements(`style:not([id^="${mount}-"],[id^="swal-pub"],[class^="darkreader"])`, function (tag) {
  1387. let id = tag.attr("id");
  1388. let text = tag.html()
  1389. if (tag.data("styles") === text) return;
  1390. tag.data("styles", text);
  1391. // 替换颜色并添加样式
  1392. let cssText = base.adaptiveStyleOverride(text, "", type, colorMap);
  1393. if (text === cssText) return;
  1394. id = id ? id : `${mount}-ColorUI-${count++}`
  1395. base.addStyle(id, 'style', cssText, tag[0], "after");
  1396. }, true)
  1397. base.waitForKeyElements('svg', function (element) {
  1398. element.find('*').each((index, element) => {
  1399. let fill = $(element).attr('fill');
  1400. let stroke = $(element).attr('stroke');
  1401. if (fill) {
  1402. let newFill = base.adaptiveStyleOverride(fill, "", type, colorMap);
  1403. if (newFill !== fill) {
  1404. $(element).attr('fill', newFill);
  1405. }
  1406. }
  1407. if (stroke) {
  1408. let newStroke = base.adaptiveStyleOverride(stroke, "", type, colorMap);
  1409. if (newStroke !== stroke) {
  1410. $(element).attr('stroke', newStroke);
  1411. }
  1412. }
  1413. });
  1414. }, true);
  1415. base.waitForKeyElements(`[style]:not([id^="${mount}-"],[class*="listener-"])`, function (element) {
  1416. if (element.attr("data-pl-colored") === temp.color) return;
  1417. let originalStyle = element.attr("style");
  1418. if (!originalStyle) return;
  1419. let newStyle = base.adaptiveStyleOverride(originalStyle, "", type, colorMap);
  1420. if (newStyle !== originalStyle) {
  1421. element.attr("style", newStyle);
  1422. element.attr("data-pl-colored", temp.color);
  1423. }
  1424. }, true);
  1425. temp.colored = true;
  1426. }
  1427. },
  1428. /**
  1429. * 延时执行
  1430. * @author 油小猴
  1431. * @description 仅可于 `async` 函数中执行,否则无法倒计时。
  1432. * @param {number} time - 等待时间(毫秒)
  1433. * @returns {Promise<void>} 延时完成的 `Promise`
  1434. */
  1435. sleep(time) {
  1436. return new Promise(resolve => setTimeout(resolve, time));
  1437. },
  1438. /**
  1439. * 判断版本号新旧
  1440. * @author hmjz100
  1441. * @description 该函数将版本号按 `.` 分割为数字数组,逐段比较大小。
  1442. * 若某段 a 的数字大于 b,则 a 更新;
  1443. * 若所有段均相等,则版本相等(返回 false)。
  1444. * @param {string} a - 待比较的版本号
  1445. * @param {string} b - 基准版本号(如 "1.0.9.7")
  1446. * @returns {boolean} - 若 a 比 b 更新,返回 true;否则返回 false
  1447. */
  1448. isNewerVersion(a, b) {
  1449. let partsA = a.split('.').map(Number);
  1450. let partsB = b.split('.').map(Number);
  1451. let maxLength = Math.max(partsA.length, partsB.length);
  1452. for (let i = 0; i < maxLength; i++) {
  1453. let numA = partsA[i] || 0;
  1454. let numB = partsB[i] || 0;
  1455. if (numA > numB) return true;
  1456. if (numA < numB) return false;
  1457. }
  1458. return false;
  1459. },
  1460. /**
  1461. * 提取版本号主版本
  1462. * @author 油小猴
  1463. * @param {string} version - 完整版本号(如 `1.2.3`)
  1464. * @returns {string|null} 主版本号(如 `1`)或 `null`(格式错误时)
  1465. */
  1466. getMajorVersion(version) {
  1467. let [major] = (version || "").split('.');
  1468. return /^\d+$/.test(major) ? major : null;
  1469. },
  1470. /**
  1471. * 查找 React 组件实例
  1472. * @author 油小猴
  1473. * @description 支持 Fiber 架构遍历,可指定向上查找层级
  1474. * @param {HTMLElement} dom - 起始 DOM 元素
  1475. * @param {number} [traverseUp=0] - 向上遍历层级
  1476. * @returns {Object|null} React 组件实例或 `null`
  1477. */
  1478. findReact(dom, traverseUp = 0) {
  1479. let key = Object.keys(dom).find(key => {
  1480. return key.startsWith("__reactFiber$")
  1481. || key.startsWith("__reactInternalInstance$");
  1482. });
  1483. let domFiber = dom[key];
  1484. if (domFiber == null) return null;
  1485. if (domFiber._currentElement) {
  1486. let compFiber = domFiber._currentElement._owner;
  1487. for (let i = 0; i < traverseUp; i++) {
  1488. compFiber = compFiber._currentElement._owner;
  1489. }
  1490. return compFiber._instance;
  1491. }
  1492. let GetCompFiber = fiber => {
  1493. let parentFiber = fiber.return;
  1494. while (base.isType(parentFiber.type) == "string") {
  1495. parentFiber = parentFiber.return;
  1496. }
  1497. return parentFiber;
  1498. };
  1499. let compFiber = GetCompFiber(domFiber);
  1500. for (let i = 0; i < traverseUp; i++) {
  1501. compFiber = GetCompFiber(compFiber);
  1502. }
  1503. return compFiber.stateNode || compFiber;
  1504. },
  1505. /**
  1506. * 迁移旧版本配置
  1507. * @author hmjz100
  1508. * @description 将旧版配置项目迁移到新版配置
  1509. */
  1510. initConfigMigration(latest) {
  1511. try {
  1512. if (latest === 1) {
  1513. let mapping = {
  1514. 'setting_rpc_domain': ['setting_aria2_rpc', 0, 'domain'],
  1515. 'setting_rpc_port': ['setting_aria2_rpc', 0, 'port'],
  1516. 'setting_rpc_path': ['setting_aria2_rpc', 0, 'path'],
  1517. 'setting_rpc_token': ['setting_aria2_rpc', 0, 'token'],
  1518. 'setting_rpc_dir': ['setting_aria2_rpc', 0, 'dir'],
  1519. 'setting_terminal_type': ['setting_curl_terminal'],
  1520. 'setting_init_code': ['setting_init', 'code'],
  1521. 'setting_init_license': ['setting_init', 'license'],
  1522. 'setting_init_version': ['setting_init', 'version'],
  1523. 'setting_theme_color': ['setting_ui_theme', 'color'],
  1524. 'setting_theme_baidu': ['setting_ui_theme', 'custom', '$baidu'],
  1525. 'setting_theme_ali': ['setting_ui_theme', 'custom', '$aliyun'],
  1526. 'setting_theme_mcloud': ['setting_ui_theme', 'custom', '$mcloud'],
  1527. 'setting_theme_tcloud': ['setting_ui_theme', 'custom', '$tcloud'],
  1528. 'setting_theme_xunlei': ['setting_ui_theme', 'custom', '$xunlei'],
  1529. 'setting_theme_quark': ['setting_ui_theme', 'custom', '$quark'],
  1530. 'setting_theme_uc': ['setting_ui_theme', 'custom', '$uc'],
  1531. 'setting_theme_123': ['setting_ui_theme', 'custom', '$123pan']
  1532. };
  1533. // 旧版配置执行迁移
  1534. for (let oldKey in mapping) {
  1535. let val = base.getValue(oldKey);
  1536. if (val === undefined || val === null) continue;
  1537. val = (val === 'no' ? false : val === 'yes' ? true : val);
  1538. let path = mapping[oldKey];
  1539. if (path.length === 1) {
  1540. base.setValue(path[0], val);
  1541. } else {
  1542. let [root, ...keys] = path;
  1543. let obj = base.getValue(root);
  1544. if (obj === undefined || obj === null) {
  1545. let firstKeyType = typeof keys[0];
  1546. let isIndex = firstKeyType === 'number' || (firstKeyType === 'string' && /^\d+$/.test(keys[0]));
  1547. obj = isIndex ? [] : {};
  1548. }
  1549. let ref = obj;
  1550. for (let i = 0; i < keys.length - 1; i++) {
  1551. let key = keys[i];
  1552. if (!ref[key]) {
  1553. let nextKey = keys[i + 1];
  1554. let hasNextIndex = nextKey !== undefined && (base.isType(nextKey === 'number' || (typeof nextKey) === 'string' && /^\d+$/.test(nextKey)));
  1555. ref[key] = hasNextIndex ? [] : {};
  1556. }
  1557. ref = ref[key];
  1558. }
  1559. ref[keys.slice(-1)[0]] = val;
  1560. base.setValue(root, obj);
  1561. }
  1562. base.delValue(oldKey);
  1563. }
  1564. }
  1565. } catch (e) {
  1566. console.error("【LinkSwift】迁移旧版本配置到新配置时出错", e);
  1567. }
  1568. },
  1569. /**
  1570. * 初始化默认配置
  1571. * @author 油小猴
  1572. * @author hmjz100
  1573. * @description 创建基础配置、主题设置等存储项(仅当不存在时)
  1574. */
  1575. initDefaultConfig() {
  1576. if (base.getValue('setting_config_version') !== '1') base.initConfigMigration(1);
  1577. // 设置新结构的默认值(仅当未设置时)
  1578. let defaults = [
  1579. {
  1580. name: "setting_aria2_rpc",
  1581. value: [
  1582. {
  1583. domain: "http://localhost",
  1584. port: "16800",
  1585. path: "/jsonrpc",
  1586. token: "",
  1587. dir: "",
  1588. default: true
  1589. }
  1590. ]
  1591. },
  1592. {
  1593. name: "setting_bitcomet_rpc",
  1594. value: [
  1595. {
  1596. domain: "http://localhost",
  1597. port: "8080",
  1598. path: "/panel/task_add_httpftp_result",
  1599. authName: "",
  1600. authPass: "",
  1601. dir: "",
  1602. default: true
  1603. }
  1604. ]
  1605. },
  1606. {
  1607. name: "setting_abdm_rpc",
  1608. value: [
  1609. {
  1610. domain: "http://localhost",
  1611. port: "15151",
  1612. dir: "",
  1613. default: true
  1614. }
  1615. ]
  1616. },
  1617. {
  1618. name: "setting_curl_terminal",
  1619. value: "wc"
  1620. },
  1621. {
  1622. name: "setting_init",
  1623. value: {
  1624. code: "",
  1625. license: "",
  1626. version: ""
  1627. }
  1628. },
  1629. {
  1630. name: "setting_ui_theme",
  1631. value: {
  1632. color: "#574AB8",
  1633. custom: {
  1634. $baidu: false,
  1635. $aliyun: false,
  1636. $mcloud: false,
  1637. $tcloud: false,
  1638. $xunlei: false,
  1639. $quark: false,
  1640. $uc: false,
  1641. $123pan: false
  1642. }
  1643. }
  1644. },
  1645. {
  1646. name: "setting_config_version",
  1647. value: "1"
  1648. }
  1649. ];
  1650. function cloneDeep(item) {
  1651. return JSON.parse(JSON.stringify(item));
  1652. }
  1653. function fillMissingFields(target, source) {
  1654. // 如果 target 不存在,直接返回 source 的深拷贝
  1655. if (target === null || target === undefined) {
  1656. return cloneDeep(source);
  1657. }
  1658. // 如果类型不同,直接替换为 source
  1659. if (typeof source !== typeof target) {
  1660. return cloneDeep(source);
  1661. }
  1662. // 如果 source 是对象
  1663. if (base.isType(source) === 'object' && !Array.isArray(source)) {
  1664. if (typeof target !== 'object' || Array.isArray(target)) {
  1665. return cloneDeep(source);
  1666. }
  1667. let result = { ...target };
  1668. for (let key in source) {
  1669. if (!source.hasOwnProperty(key)) continue;
  1670. // 跳过 default 的自动合并
  1671. if (key === 'default') continue;
  1672. if (key === 'dir' && target[key] !== undefined) continue;
  1673. if (key === 'token' && target[key] !== undefined) continue;
  1674. if (key === 'authName' && target[key] !== undefined) continue;
  1675. if (key === 'authPass' && target[key] !== undefined) continue;
  1676. result[key] = fillMissingFields(target[key], source[key]);
  1677. }
  1678. return result;
  1679. }
  1680. // 如果 source 是数组
  1681. if (Array.isArray(source)) {
  1682. if (!Array.isArray(target)) {
  1683. return cloneDeep(source);
  1684. }
  1685. let result = [...target];
  1686. if (source.length > 0 && base.isType(source[0]) === 'object' && source[0] !== null) {
  1687. let template = source[0];
  1688. // 填充字段
  1689. for (let i = 0; i < result.length; i++) {
  1690. if (base.isType(result[i]) === 'object' && result[i] !== null) {
  1691. result[i] = fillMissingFields(result[i], template);
  1692. } else {
  1693. result[i] = cloneDeep(template);
  1694. }
  1695. }
  1696. // 自动补充 default: true
  1697. if (
  1698. template.default === true &&
  1699. !result.some(item => item && item.default === true) &&
  1700. result.length > 0
  1701. ) {
  1702. result[0].default = true;
  1703. }
  1704. }
  1705. return result;
  1706. }
  1707. // 基本类型,保留原始值
  1708. return target;
  1709. }
  1710. defaults.forEach(({ name, value }) => {
  1711. let current = base.getValue(name);
  1712. if (
  1713. current === null ||
  1714. current === undefined ||
  1715. (Array.isArray(current) && current.length === 0)
  1716. ) {
  1717. base.setValue(name, cloneDeep(value));
  1718. } else {
  1719. base.setValue(name, fillMissingFields(current, value));
  1720. }
  1721. });
  1722. },
  1723. /**
  1724. * 显示设置界面
  1725. * @author 油小猴
  1726. * @author hmjz100
  1727. * @description 构建包含 RPC 配置、终端类型等设置项的交互界面
  1728. * @see {@link https://www.youxiaohou.com/zh-cn/motrix.html#使用指南 RPC 配置说明}、 {@link https://www.youxiaohou.com/zh-cn/curl.html cURL 使用教程}
  1729. */
  1730. showSetting(event) {
  1731. let setting = $(`<div>
  1732. <div style="text-align:center;">带星号的设置项目将在网页刷新后生效</div>
  1733. <label class="pl-setting-item listener-tip aria2" data-title="有关 Aria2 远程服务的配置">
  1734. <div>Aria2 服务器</div>
  1735. <button type="button" class="pl-button-mini swal2-confirm swal2-styled listener-open-aria2-setting" data-back-to-setting="true"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg><span>配置</span></button>
  1736. </label>
  1737. <label class="pl-setting-item listener-tip bitcomet" data-title="有关比特彗星远程服务的配置">
  1738. <div>比特彗星服务器</div>
  1739. <button type="button" class="pl-button-mini swal2-confirm swal2-styled listener-open-bitcomet-setting" data-back-to-setting="true"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg><span>配置</span></button>
  1740. </label>
  1741. <label class="pl-setting-item listener-tip abdm" data-title="有关 AB Download Manager 远程服务的配置">
  1742. <div>AB Download Manager 服务器</div>
  1743. <button type="button" class="pl-button-mini swal2-confirm swal2-styled listener-open-abdm-setting" data-back-to-setting="true"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg><span>配置</span></button>
  1744. </label>
  1745. <label class="pl-setting-item curl">
  1746. <div>终端类型</div>
  1747. <select class="swal2-select pl-input listener-terminal">
  1748. ${Object.keys(temp.terminalType).map(i => `<option value="${i}" ${base.getValue('setting_curl_terminal') === i ? 'selected' : ""}>${temp.terminalType[i]}</option>`).join("")}
  1749. </select>
  1750. </label>
  1751. <div class="curl" style="display:flex;justify-content:flex-end;"><a href="https://www.youxiaohou.com/zh-cn/curl.html" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg> cURL使用教程</a>,适用于 cURL 下载👆</div>
  1752. <div class="other" style="display:flex;justify-content:center;margin-top:20px"><button type="button" class="pl-button-mini swal2-deny swal2-styled listener-unregister listener-tip" data-title="仅会清除已存储的百度令牌,其余设置项目无影响,仍会保留"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg><span>熄灭已经点亮的按钮*</span></button></div>
  1753. </div>`);
  1754. Swal.fire({
  1755. ...temp.swalDefault,
  1756. title: '(。•ᴗ•。) 助手设置',
  1757. html: setting.html(),
  1758. icon: 'info',
  1759. iconHtml: '⚙︎',
  1760. allowOutsideClick: false,
  1761. showCloseButton: true,
  1762. showConfirmButton: false,
  1763. footer: `<p><a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;&#47;&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;" target="_blank" class="pl-a"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg>&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;</a>&#32;&#30001;&#32;<a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;" target="_blank" class="pl-a">&#104;&#109;&#106;&#122;&#49;&#48;&#48;</a>&#32;&#21046;&#20316;</p><p>${config.base.dom.footer}</p>`,
  1764. didOpen: (toast) => {
  1765. let element = $(toast);
  1766. if (event && Object.keys($(event.currentTarget).data()).some(key => key.startsWith('backTo'))) element.find('.swal2-close').addClass('listener-tip').attr('data-title', '返回上页').css({ "left": "0", "right": "auto" }).text("◃");
  1767. if (event && $(event.currentTarget).data("back-to-downloads")) element.find(".aria2, .bitcomet, .abdm, .other").hide();
  1768. },
  1769. willClose: () => {
  1770. if (event && $(event.currentTarget).data("back-to-downloads")) base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  1771. },
  1772. });
  1773. },
  1774. /**
  1775. * 显示 Aria2 服务设置界面
  1776. * @author hmjz100
  1777. * @description 包含 RPC 配置的交互界面
  1778. * @see {@link https://www.youxiaohou.com/zh-cn/motrix.html#使用指南 RPC 配置说明}
  1779. */
  1780. showAria2Setting(event) {
  1781. let AriaList = base.getValue('setting_aria2_rpc');
  1782. let AriaOptions = AriaList.map((item, index) => {
  1783. return `<option value="${index}"${item.default ? ' selected' : ""}>${item.domain ? item.domain : ""}:${item.port ? item.port : ""}${item.path ? item.path : ""}</option>`;
  1784. }).join("");
  1785. let AriaSelected = AriaList.find(i => i.default);
  1786. let Aria2Setting = `<div style="text-align:center;"><a href="https://www.youxiaohou.com/zh-cn/motrix.html#使用指南" target="_blank" class="pl-a" data-no-instant="1"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-link"></use></svg> RPC配置说明</a>,适用于 Aria2 推送下载</div>
  1787. <label class="pl-setting-item">
  1788. <div>默认配置</div>
  1789. <div>
  1790. <select class="swal2-select pl-input listener-rpc-select" data-type="aria2" style="max-width:50%;min-width:auto">
  1791. ${AriaOptions}<option value="new">+ 创建新项目</option>
  1792. </select>
  1793. <button type="button" class="pl-button-mini swal2-deny swal2-styled listener-rpc-delete" data-type="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg><span>删除</span></button>
  1794. <button type="button" class="pl-button-mini swal2-confirm swal2-styled listener-rpc-test" data-type="aria2" style="margin-left:0"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg><span>测试</span></button>
  1795. </div>
  1796. </label>
  1797. <label class="pl-setting-item">
  1798. <div>服务主机</div>
  1799. <input type="text" autocomplete="off" placeholder="主机名,仅支持以及需带上 http(s)://,不支持 ws" class="swal2-input pl-input listener-rpc-input" data-type="aria2.domain" value="">
  1800. </label>
  1801. <label class="pl-setting-item">
  1802. <div>服务端口</div>
  1803. <input type="text" autocomplete="off" placeholder="端口号,例如 Motrix 为 16800,Aria2 为 6800" class="swal2-input pl-input listener-rpc-input" data-type="aria2.port" value="">
  1804. </label>
  1805. <label class="pl-setting-item">
  1806. <div>服务路径</div>
  1807. <input type="text" autocomplete="off" placeholder="访问路径,一般是 /jsonrpc" class="swal2-input pl-input listener-rpc-input" data-type="aria2.path" value="">
  1808. </label>
  1809. <label class="pl-setting-item">
  1810. <div>服务密钥</div>
  1811. <input type="text" autocomplete="off" placeholder="无密钥无需填写" class="swal2-input pl-input listener-rpc-input" data-type="aria2.token" value="">
  1812. </label>
  1813. <label class="pl-setting-item">
  1814. <div>存储路径</div>
  1815. <input type="text" autocomplete="off" placeholder="文件下载后保存路径,例如 D:\\Downloads\\,留空则默认" class="swal2-input pl-input listener-rpc-input" data-type="aria2.dir" value="">
  1816. </label>`;
  1817. Swal.fire({
  1818. ...temp.swalDefault,
  1819. title: 'Aria2 服务设置',
  1820. html: Aria2Setting,
  1821. icon: 'info',
  1822. iconHtml: '⚙︎',
  1823. allowOutsideClick: false,
  1824. showCloseButton: true,
  1825. showConfirmButton: false,
  1826. footer: `<p><a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;&#47;&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;" target="_blank" class="pl-a"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg>&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;</a>&#32;&#30001;&#32;<a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;" target="_blank" class="pl-a">&#104;&#109;&#106;&#122;&#49;&#48;&#48;</a>&#32;&#21046;&#20316;</p><p>${config.base.dom.footer}</p>`,
  1827. didOpen: (toast) => {
  1828. let element = $(toast);
  1829. if (event && Object.keys($(event.currentTarget).data()).some(key => key.startsWith('backTo'))) element.find('.swal2-close').addClass('listener-tip').attr('data-title', '返回上页').css({ "left": "0", "right": "auto" }).text("◃");
  1830. if (AriaSelected) {
  1831. element.find('.listener-rpc-input').each(function () {
  1832. let type = $(this).data('type').split(".")[1];
  1833. $(this).val(AriaSelected[type] || "");
  1834. });
  1835. } else {
  1836. AriaList[0].default = true;
  1837. base.setValue('setting_aria2_rpc', AriaList);
  1838. AriaSelected = AriaList[0];
  1839. }
  1840. },
  1841. willClose: () => {
  1842. if (event && $(event.currentTarget).data("back-to-setting")) base.showSetting();
  1843. if (event && $(event.currentTarget).data("back-to-downloads")) base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  1844. },
  1845. });
  1846. },
  1847. /**
  1848. * 显示比特彗星服务设置界面
  1849. * @author hmjz100
  1850. * @description 包含 RPC 配置的交互界面
  1851. */
  1852. showBitcometSetting(event) {
  1853. let BCList = base.getValue('setting_bitcomet_rpc');
  1854. let BCOptions = BCList.map((item, index) => {
  1855. return `<option value="${index}"${item.default ? ' selected' : ""}>${item.domain ? item.domain : ""}:${item.port ? item.port : ""}${item.path ? item.path : ""}</option>`;
  1856. }).join("");
  1857. let BCSelected = BCList.find(i => i.default);
  1858. let BitcometSetting = `<div style="text-align:center;">适用于比特彗星推送下载</div>
  1859. <label class="pl-setting-item">
  1860. <div>默认配置</div>
  1861. <div>
  1862. <select class="swal2-select pl-input listener-rpc-select" data-type="bitcomet" style="max-width:75%;min-width:auto">
  1863. ${BCOptions}<option value="new">+ 创建新项目</option>
  1864. </select>
  1865. <button type="button" class="pl-button-mini swal2-deny swal2-styled listener-rpc-delete" data-type="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg><span>删除</span></button>
  1866. </div>
  1867. </label>
  1868. <label class="pl-setting-item">
  1869. <div>服务主机</div>
  1870. <input type="text" autocomplete="off" placeholder="主机地址,需带上 http(s)://,但不需要写端口与路径" class="swal2-input pl-input listener-rpc-input" data-type="bitcomet.domain" value="">
  1871. </label>
  1872. <label class="pl-setting-item">
  1873. <div>服务端口</div>
  1874. <input type="text" autocomplete="off" placeholder="服务器端口号,一般为 8080" class="swal2-input pl-input listener-rpc-input" data-type="bitcomet.port" value="">
  1875. </label>
  1876. <label class="pl-setting-item">
  1877. <div>服务路径</div>
  1878. <input type="text" autocomplete="off" placeholder="一般是 /panel/task_add_httpftp_result" class="swal2-input pl-input listener-rpc-input" data-type="bitcomet.path" value="">
  1879. </label>
  1880. <label class="pl-setting-item">
  1881. <div>服务账号</div>
  1882. <input type="text" autocomplete="off" placeholder="本地服务器账号,无账号无需填写" class="swal2-input pl-input listener-rpc-input" data-type="bitcomet.authName" value="">
  1883. </label>
  1884. <label class="pl-setting-item">
  1885. <div>服务密码</div>
  1886. <input type="text" autocomplete="off" placeholder="本地服务器密码,无密码无需填写" class="swal2-input pl-input listener-rpc-input" data-type="bitcomet.authPass" value="">
  1887. </label>
  1888. <label class="pl-setting-item">
  1889. <div>存储路径</div>
  1890. <input type="text" autocomplete="off" placeholder="文件下载后保存路径,例如 D:\\Downloads\\,留空则默认" class="swal2-input pl-input listener-rpc-input" data-type="bitcomet.dir" value="">
  1891. </label>`;
  1892. Swal.fire({
  1893. ...temp.swalDefault,
  1894. title: '比特彗星服务设置',
  1895. html: BitcometSetting,
  1896. icon: 'info',
  1897. iconHtml: '⚙︎',
  1898. allowOutsideClick: false,
  1899. showCloseButton: true,
  1900. showConfirmButton: false,
  1901. footer: `<p><a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;&#47;&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;" target="_blank" class="pl-a"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg>&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;</a>&#32;&#30001;&#32;<a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;" target="_blank" class="pl-a">&#104;&#109;&#106;&#122;&#49;&#48;&#48;</a>&#32;&#21046;&#20316;</p><p>${config.base.dom.footer}</p>`,
  1902. didOpen: (toast) => {
  1903. let element = $(toast);
  1904. if (event && Object.keys($(event.currentTarget).data()).some(key => key.startsWith('backTo'))) element.find('.swal2-close').addClass('listener-tip').attr('data-title', '返回上页').css({ "left": "0", "right": "auto" }).text("◃");
  1905. if (BCSelected) {
  1906. element.find('.listener-rpc-input').each(function () {
  1907. let type = $(this).data('type').split(".")[1];
  1908. $(this).val(BCSelected[type] || "");
  1909. });
  1910. } else {
  1911. BCSelected[0].default = true;
  1912. base.setValue('setting_bitcomet_rpc', BCSelected);
  1913. BCSelected = BCList[0];
  1914. }
  1915. },
  1916. willClose: () => {
  1917. if (event && $(event.currentTarget).data("back-to-setting")) base.showSetting();
  1918. if (event && $(event.currentTarget).data("back-to-downloads")) base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  1919. },
  1920. });
  1921. },
  1922. /**
  1923. * 显示 AB Download Manager 服务设置界面
  1924. * @author hmjz100
  1925. * @description 包含 RPC 配置的交互界面
  1926. */
  1927. showABDMSetting(event) {
  1928. let ABList = base.getValue('setting_abdm_rpc');
  1929. let ABOptions = ABList.map((item, index) => {
  1930. return `<option value="${index}"${item.default ? ' selected' : ""}>${item.domain}:${item.port}</option>`;
  1931. }).join("");
  1932. let ABSelected = ABList.find(i => i.default);
  1933. let ABSetting = `<div style="text-align:center;">适用于 AB Download Manager 推送下载</div>
  1934. <label class="pl-setting-item">
  1935. <div>默认配置</div>
  1936. <div>
  1937. <select class="swal2-select pl-input listener-rpc-select" data-type="abdm" style="max-width:50%;min-width:auto">
  1938. ${ABOptions}<option value="new">+ 创建新项目</option>
  1939. </select>
  1940. <button type="button" class="pl-button-mini swal2-deny swal2-styled listener-rpc-delete" data-type="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg><span>删除</span></button>
  1941. <button type="button" class="pl-button-mini swal2-confirm swal2-styled listener-rpc-test" data-type="abdm" style="margin-left:0"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg><span>测试</span></button>
  1942. </div>
  1943. </label>
  1944. <label class="pl-setting-item">
  1945. <div>服务主机</div>
  1946. <input type="text" autocomplete="off" placeholder="主机地址,需带上 http(s)://,但不需要写端口与路径" class="swal2-input pl-input listener-rpc-input" data-type="abdm.domain" value="">
  1947. </label>
  1948. <label class="pl-setting-item">
  1949. <div>服务端口</div>
  1950. <input type="text" autocomplete="off" placeholder="服务器端口号,一般为 15151" class="swal2-input pl-input listener-rpc-input" data-type="abdm.port" value="">
  1951. </label>
  1952. <label class="pl-setting-item">
  1953. <div>存储路径</div>
  1954. <input type="text" autocomplete="off" placeholder="文件下载后保存路径,例如 D:\\Downloads\\,留空则默认" class="swal2-input pl-input listener-rpc-input" data-type="abdm.dir" value="">
  1955. </label>`;
  1956. Swal.fire({
  1957. ...temp.swalDefault,
  1958. title: 'ABDM 服务设置',
  1959. html: ABSetting,
  1960. icon: 'info',
  1961. iconHtml: '⚙︎',
  1962. allowOutsideClick: false,
  1963. showCloseButton: true,
  1964. showConfirmButton: false,
  1965. footer: `<p><a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;&#47;&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;" target="_blank" class="pl-a"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg>&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;</a>&#32;&#30001;&#32;<a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;" target="_blank" class="pl-a">&#104;&#109;&#106;&#122;&#49;&#48;&#48;</a>&#32;&#21046;&#20316;</p><p>${config.base.dom.footer}</p>`,
  1966. didOpen: (toast) => {
  1967. let element = $(toast);
  1968. if (event && Object.keys($(event.currentTarget).data()).some(key => key.startsWith('backTo'))) element.find('.swal2-close').addClass('listener-tip').attr('data-title', '返回上页').css({ "left": "0", "right": "auto" }).text("◃");
  1969. if (ABSelected) {
  1970. element.find('.listener-rpc-input').each(function () {
  1971. let type = $(this).data('type').split(".")[1];
  1972. $(this).val(ABSelected[type] || "");
  1973. });
  1974. } else {
  1975. ABSelected[0].default = true;
  1976. base.setValue('setting_abdm_rpc', ABSelected);
  1977. ABSelected = BCList[0];
  1978. }
  1979. },
  1980. willClose: () => {
  1981. if (event && $(event.currentTarget).data("back-to-setting")) base.showSetting();
  1982. if (event && $(event.currentTarget).data("back-to-downloads")) base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  1983. },
  1984. });
  1985. },
  1986. /**
  1987. * 显示美化设置界面
  1988. * @author hmjz100
  1989. * @description 提供主题颜色选择器和各网盘界面美化配置
  1990. * @fires .listener-color - 主题色选择事件
  1991. * @fires .listener-theme - 网盘主题配置变更事件
  1992. */
  1993. showBeautify() {
  1994. function changeColor() {
  1995. temp.color = base.getValue('setting_ui_theme').color;
  1996. return config.base.dom.theme.map(item => {
  1997. return `<div style="--color:${item.color}" class="listener-color" data-color="${item.color}">
  1998. <div class="mask">
  1999. ${item.name.split('|').map(part => `<div>${part}</div>`).join("")}
  2000. ${item.color === temp.color ? '<div class="this"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"></use></svg></div>' : ""}
  2001. </div>
  2002. </div>`;
  2003. }).join("")
  2004. }
  2005. function changeTheme() {
  2006. let themeList = [
  2007. { name: '百度网盘', key: '$baidu' },
  2008. { name: '阿里云盘', key: '$aliyun' },
  2009. { name: '移动云盘', key: '$mcloud' },
  2010. { name: '天翼云盘', key: '$tcloud' },
  2011. { name: '迅雷云盘', key: '$xunlei' },
  2012. { name: '夸克网盘', key: '$quark' },
  2013. { name: 'UC 网盘', key: '$uc' },
  2014. { name: '123 云盘', key: '$123pan' }
  2015. ];
  2016. return themeList.map(item => {
  2017. return `<label class="pl-setting-item">
  2018. <div>${item.name}</div>
  2019. <input type="checkbox" class="swal2-checkbox pl-input listener-theme" data-type="${item.key}" ${base.getValue('setting_ui_theme').custom[item.key] === true ? 'checked' : ""}>
  2020. </label>`;
  2021. }).join("");
  2022. }
  2023. let beautify = $(`<div>
  2024. <div style="text-align:center;">带星号的美化项目将在网页刷新后生效</div>
  2025. <label class="pl-setting-item" style="justify-content:center"><div class="pl-color">${changeColor()}</div></label>
  2026. <div class="pl-setting-item"><div>替换界面配色为主题颜色*</div><div class="pl-checkboxies">${changeTheme()}</div></div>
  2027. <style>
  2028. .pl-color{display:grid!important;grid-template-columns:repeat(5, var(--pl-color-width));gap:10px;--pl-color-width:55px}
  2029. .pl-color > div{background-color:var(--color);width:var(--pl-color-width);height:var(--pl-color-width);box-sizing:border-box;cursor:pointer}
  2030. .pl-color .mask{width:calc(var(--pl-color-width) - 2px);height:calc(var(--pl-color-width) - 2px);opacity:0;transition:opacity.2s;color:#fff;font-size:13px;display:flex;align-items:center;justify-content:center;flex-direction:column}
  2031. .pl-color > div:hover .mask{opacity:1}
  2032. .pl-checkboxies{display:grid!important;grid-template-columns:repeat(2, 98px);gap:10px}
  2033. .pl-input[type=checkbox]{height:20px;width:20px;padding:0!important;background-image:none!important}
  2034. </style>
  2035. </div>`)
  2036. Swal.fire({
  2037. ...temp.swalDefault,
  2038. title: '(✿ᴗ‿ᴗ) 助手美化',
  2039. html: beautify.html(),
  2040. icon: 'success',
  2041. iconHtml: '🍃︎',
  2042. allowOutsideClick: false,
  2043. showCloseButton: true,
  2044. showConfirmButton: false,
  2045. footer: `<p><a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;&#47;&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;" target="_blank" class="pl-a"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg>&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;</a>&#32;&#30001;&#32;<a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;" target="_blank" class="pl-a">&#104;&#109;&#106;&#122;&#49;&#48;&#48;</a>&#32;&#21046;&#20316;</p><p>${config.base.dom.footer}</p>`,
  2046. });
  2047. },
  2048. /**
  2049. * 显示调试信息面板
  2050. * @description 展示脚本运行时环境、版本信息及依赖状态
  2051. * @author hmjz100
  2052. * @property{string} manageHandler - 外部管理器名称
  2053. * @property{string} manageVersion - 外部管理器版本
  2054. */
  2055. showDebug() {
  2056. let debugInfo = "";
  2057. debugInfo += `<span>以下内容均为脚本自检信息<br/>本页面仅作为调试使用<span>`;
  2058. debugInfo += `<label class="pl-setting-item"><div>[外] 管理器名称</div>${info.mhandler ? info.mhandler : "无法获取"}</label>`;
  2059. debugInfo += `<label class="pl-setting-item"><div>[外] 管理器版本</div>${info.mversion ? info.mversion : "无法获取"}</label>`;
  2060. debugInfo += `<label class="pl-setting-item"><div>[内] 脚本挂载点</div>${mount ? `${mount.toLowerCase()}.${mount}` : "无法获取"}</label>`;
  2061. debugInfo += `<label class="pl-setting-item"><div>[外] 脚本名称</div>${info.name ? info.name : "无法获取"}</label>`;
  2062. debugInfo += `<label class="pl-setting-item"><div>[外] 脚本作者</div>${info.author ? info.author : "无法获取"}</label>`;
  2063. debugInfo += `<label class="pl-setting-item"><div>[外/内] 脚本版本</div>${info.version ? info.version : "无法获取"}</label>`;
  2064. debugInfo += `<label class="pl-setting-item"><div>[外/内] 脚本图标</div>${info.icon ? `<img style="max-width:30%" src="${info.icon}"></img>` : "无法获取"}</label>`;
  2065. debugInfo += `<label class="pl-setting-item"><div>[内] 公众号二维码</div>${config.base?.service?.account ? `<img style="max-width:30%" src="${config.base.service.account}"></img>` : "无法获取"}</label>`;
  2066. debugInfo = '<div>' + debugInfo + '</div>';
  2067. Swal.fire({
  2068. ...temp.swalDefault,
  2069. icon: 'info',
  2070. title: '调试信息',
  2071. html: debugInfo,
  2072. allowOutsideClick: false,
  2073. showCloseButton: true,
  2074. footer: `<p><a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;&#47;&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;" target="_blank" class="pl-a"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg>&#76;&#105;&#110;&#107;&#83;&#119;&#105;&#102;&#116;</a>&#32;&#30001;&#32;<a href="&#104;&#116;&#116;&#112;&#115;&#58;&#47;&#47;&#103;&#105;&#116;&#104;&#117;&#98;&#46;&#99;&#111;&#109;&#47;&#104;&#109;&#106;&#122;&#49;&#48;&#48;" target="_blank" class="pl-a">&#104;&#109;&#106;&#122;&#49;&#48;&#48;</a>&#32;&#21046;&#20316;</p><p>${config.base.dom.footer}</p>`,
  2075. });
  2076. },
  2077. /**
  2078. * 显示版本更新日志
  2079. * @author hmjz100
  2080. * @description 按时间倒序展示所有历史版本更新内容
  2081. */
  2082. async showUpdate() {
  2083. await Swal.fire({
  2084. ...temp.swalDefault,
  2085. icon: 'info',
  2086. title: '更新日志',
  2087. html: `<div class="version-log">
  2088. <div class="block">
  2089. <blockquote>
  2090. <div>风雨送春归,飞雪迎春到。已是悬崖百丈冰,犹有花枝俏。</div>
  2091. <div>俏也不争春,只把春来报。待到山花烂漫时,她在丛中笑。</div>
  2092. </blockquote>
  2093. </div>
  2094. <div class="block">(ノ◕ヮ◕)ノ 遇到 Bug 要记得去 <a class="pl-a" href="https://github.com/hmjz100/LinkSwift/issues" target="_blank">Github 议题</a> 向我报告哦~</div>
  2095. <div class="block">
  2096. <name>V1.1.1.7</name>
  2097. <div>
  2098. <div>1、修复 - 缺失声明 @connect 导致的问题。</div>
  2099. </div>
  2100. </div>
  2101. <div class="block">
  2102. <name>V1.1.1.6</name>
  2103. <div>
  2104. <div>1、<span style="color:#fff;background:${temp.color}">废弃 - 百度网盘 BDUSS Cookie 相关代码,转向使用更安全的 AccessToken</span>;</div>
  2105. <div>2、废弃 - 百度网盘分享页面下载相关代码;</div>
  2106. <div>3、优化 - 下载窗口可在设置改变后动态修改界面。</div>
  2107. </div>
  2108. </div>
  2109. <div class="block">
  2110. <name>V1.1.1.5</name>
  2111. <div>
  2112. <div>1、新增 - AB Download Manager 下载方式;</div>
  2113. <div>2、优化 - 支持从设置页面一键返回下载窗口,无需重复获取链接。</div>
  2114. </div>
  2115. </div>
  2116. <div class="block">
  2117. <name>V1.1.1.4</name>
  2118. <div>
  2119. <div>1、适配 123 云盘新版页面。</div>
  2120. </div>
  2121. </div>
  2122. <div class="block">
  2123. <name>V1.1.1.3</name>
  2124. <div>
  2125. <div>1、修复夸克网盘无法获取下载链接的 Bug;</div>
  2126. <div>2、修复 API 下载无法复制全部链接。</div>
  2127. </div>
  2128. </div>
  2129. <div class="block">
  2130. <name>V1.1.1.2</name>
  2131. <div>
  2132. <div>1、修复无法删除第一项远程配置的 Bug。</div>
  2133. </div>
  2134. </div>
  2135. <div class="block">
  2136. <name>V1.1.1.1</name>
  2137. <div>
  2138. <div>1、修复推送到 Aria2 时推送成功但报错的 Bug。</div>
  2139. </div>
  2140. </div>
  2141. <div class="block">
  2142. <name>V1.1.1</name>
  2143. <div>
  2144. <div>1、配置文件格式更新,<span style="color:#fff;background:${temp.color}">支持添加、删除、切换多个服务配置</span>;</div>
  2145. <div>2、支持比特彗星推送下载,<span style="color:#fff;background:${temp.color}">原 RPC 已并入 Aria2 下载</span>;</div>
  2146. <div>3、界面<span style="color:#fff;background:${temp.color}">增加 Font Awesome 图标!</span>更好看啦;</div>
  2147. <div>4、优化脚本代码、界面,运行更轻快;</div>
  2148. <div>5、修复上个版本遗存的问题。</div>
  2149. </div>
  2150. </div>
  2151. <hr/>
  2152. <div class="block">
  2153. <name>V1.1.0.1</name>
  2154. <div>
  2155. <div>1、修复查看 RPC 下载任务的 Bug。</div>
  2156. </div>
  2157. </div>
  2158. <div class="block">
  2159. <name>V1.1.0</name>
  2160. <div>
  2161. <div>1、支持 UC 网盘、123 云盘;</div>
  2162. <div>2、改进了网盘主题的注入方式;</div>
  2163. <div>3、聚合并重构了部分重复函数,对整体脚本逻辑进行了梳理和精简;</div>
  2164. <div>4、将脚本执行阶段从 document-body 适配为 document-start。</div>
  2165. </div>
  2166. </div>
  2167. <hr/>
  2168. <div class="block">
  2169. <name>V1.0.9.7</name>
  2170. <div>
  2171. <div>1、修复移动云盘下载错误;</div>
  2172. <div>2、优化代码,更好的错误识别;</div>
  2173. <div>3、去除了游小猴云服务。</div>
  2174. </div>
  2175. </div>
  2176. <div class="block">
  2177. <name>V1.0.9.6</name>
  2178. <div>
  2179. <div>1、支持在百度网盘中选择文件夹下载;</div>
  2180. <div>2、优化部分提示。</div>
  2181. </div>
  2182. </div>
  2183. <div class="block">
  2184. <name>V1.0.9.5</name>
  2185. <div>
  2186. <div>1、修复因代码逻辑错误而无法获取链接的 Bug。</div>
  2187. </div>
  2188. </div>
  2189. <div class="block">
  2190. <name>V1.0.9.4</name>
  2191. <div>
  2192. <div>1、修复因百度网盘 AccessToken 过期导致无法获取链接的 Bug。</div>
  2193. </div>
  2194. </div>
  2195. <div class="block">
  2196. <name>V1.0.9.3</name>
  2197. <div>
  2198. <div>1、若网盘不支持在分享中下载,将仅显示保存网盘按钮;</div>
  2199. <div>2、优化下载界面,支持选择 Iframe Blob 的方式来下载文件,增加按钮的提示文本;</div>
  2200. <div>3、优化 CSS 样式,统一了 SweetAlert2 按钮样式,同时适配了 Dark Reader 插件,界面更协调;</div>
  2201. <div>4、支持修改油小猴网站主题色;</div>
  2202. <div>5、原有主题相关设置现已移动至助手美化页面中。</div>
  2203. </div>
  2204. </div>
  2205. <div class="block">
  2206. <name>V1.0.9.2</name>
  2207. <div>
  2208. <div>1、修复使用API 下载时有可能会导致IDM无限弹窗的Bug。</div>
  2209. </div>
  2210. </div>
  2211. <div class="block">
  2212. <name>V1.0.9.1</name>
  2213. <div>
  2214. <div>1、修复在百度网盘旧版下脚本无法删除元素的Bug。</div>
  2215. </div>
  2216. </div>
  2217. <div class="block">
  2218. <name>V1.0.9</name>
  2219. <div>
  2220. <div>1、跟进官方V6.2.7,修复因无法进行百度授权而导致获取直链报错 9019 Bug。</div>
  2221. </div>
  2222. </div>
  2223. <hr/>
  2224. <div class="block">
  2225. <name>V1.0.8.9</name>
  2226. <div>
  2227. <div>1、跟进官方V6.2.3,优化保存到网盘提示,修复阿里云盘、移动云盘失效的问题;</div>
  2228. <div>2、优化修改网盘主题的代码,减少对页面的破坏。</div>
  2229. </div>
  2230. </div>
  2231. <div class="block">
  2232. <name>V1.0.8.8</name>
  2233. <div>
  2234. <div>1、修复下载菜单字体过小的Bug。</div>
  2235. </div>
  2236. </div>
  2237. <div class="block">
  2238. <name>V1.0.8.7</name>
  2239. <div>
  2240. <div>1、修复在阿里云盘分享页面下点击“未点亮”按钮时没有任何反应的Bug;</div>
  2241. <div>2、更新并优化网盘界面精简规则;</div>
  2242. <div>3、支持更换 百度网盘、阿里云盘、迅雷云盘、夸克网盘、移动云盘 界面的主题颜色。</div>
  2243. </div>
  2244. </div>
  2245. <div class="block">
  2246. <name>V1.0.8.6</name>
  2247. <div>
  2248. <div>1、新增移动云盘会员中心页面,可在网盘中点击“会员中心”按钮查看(但无法使用第三方支付)。</div>
  2249. </div>
  2250. </div>
  2251. <div class="block">
  2252. <name>V1.0.8.5</name>
  2253. <div>
  2254. <div>1、跟进官方V6.1.6,修复迅雷网盘分享页面无法选中文件,修复移动云盘无法判断页面。</div>
  2255. </div>
  2256. </div>
  2257. <div class="block">
  2258. <name>V1.0.8.4</name>
  2259. <div>
  2260. <div>1、修复因重复绑定按钮而导致命令重复执行的Bug;</div>
  2261. <div>2、优化调试信息界面排版;</div>
  2262. <div>3、移除对百度网盘手机网页版的支持。</div>
  2263. </div>
  2264. </div>
  2265. <div class="block">
  2266. <name>V1.0.8.3</name>
  2267. <div>
  2268. <div>1、适配阿里云盘新域名alipan.com。</div>
  2269. </div>
  2270. </div>
  2271. <div class="block">
  2272. <name>V1.0.8.2</name>
  2273. <div>
  2274. <div>1、更换新图标。</div>
  2275. </div>
  2276. </div>
  2277. <div class="block">
  2278. <name>V1.0.8.1</name>
  2279. <div>
  2280. <div>1、修复因重复绑定按钮而导致 RPC 下载会发送多条下载请求的Bug;</div>
  2281. <div>2、选择不使用油小猴服务器时,“用ghproxy连接Github仓库”更换为“用jsdelivr连接Github仓库”;</div>
  2282. <div>3、跟进官方V6.1.4版本,修复移动网盘无法获取链接,支持阿里云盘新域名alipan.com。</div>
  2283. </div>
  2284. </div>
  2285. <hr/>
  2286. <div class="block">
  2287. <name>V1.0.8</name>
  2288. <div>
  2289. <div>1、修复迅雷网盘无法勾选文件。</div>
  2290. </div>
  2291. </div>
  2292. <div class="block">
  2293. <name>V1.0.7.9</name>
  2294. <div>
  2295. <div>1、更新精简网盘元素匹配规则,防止因通知横条而导致不能点到“API 下载”以下的按钮。</div>
  2296. </div>
  2297. </div>
  2298. <div class="block">
  2299. <name>V1.0.7.8</name>
  2300. <div>
  2301. <div>1、跟进官方V6.1.2,加入V2接口。</div>
  2302. <div>2、修复百度网盘下载时因为获取不到accessToken而一直转圈。</div>
  2303. </div>
  2304. </div>
  2305. <div class="block">
  2306. <name>V1.0.7.7</name>
  2307. <div>
  2308. <div>1、修复百度网盘的按钮会因为主题不同而被改变颜色的Bug;</div>
  2309. <div>2、更新夸克网盘按钮与界面。</div>
  2310. </div>
  2311. </div>
  2312. <div class="block">
  2313. <name>V1.0.7.6</name>
  2314. <div>
  2315. <div>1、修复“注入”功能;</div>
  2316. <div>2、黑暗模式支持随设置热切换。</div>
  2317. </div>
  2318. </div>
  2319. <div class="block">
  2320. <name>V1.0.7.5</name>
  2321. <div>
  2322. <div>1、修复阿里云盘下载逻辑;</div>
  2323. <div>2、精简代码;</div>
  2324. <div>3、支持深色模式;</div>
  2325. <div>4、修改部分提示文本;</div>
  2326. <div>5、修改部分CSS;</div>
  2327. <div>6、设置可测试RPC连接。</div>
  2328. </div>
  2329. </div>
  2330. <div class="block">
  2331. <name>V1.0.7.4</name>
  2332. <div>
  2333. <div>1、优化下载逻辑;</div>
  2334. <div>2、修复阿里云盘无法使用API 下载。</div>
  2335. </div>
  2336. </div>
  2337. <div class="block">
  2338. <name>V1.0.7.3</name>
  2339. <div>
  2340. <div>1、如果出现网络请求错误时支持自动重新请求;</div>
  2341. <div>2、可选择是否使用油小猴服务器。</div>
  2342. </div>
  2343. </div>
  2344. <div class="block">
  2345. <name>V1.0.7.2</name>
  2346. <div>
  2347. <div>1、修复使用 RPC 下载时会重复发送链接的Bug。</div>
  2348. </div>
  2349. </div>
  2350. <div class="block">
  2351. <name>V1.0.7.1</name>
  2352. <div>
  2353. <div>[实验功能,不影响正常使用]支持百度网盘手机网页版,勾选文件后可在顶栏找到“下载助手”按钮。</div>
  2354. </div>
  2355. </div>
  2356. <div class="block">
  2357. <name>V1.0.7</name>
  2358. <div>
  2359. <div>1、重构夸克网盘、阿里云盘按钮。</div>
  2360. </div>
  2361. </div>
  2362. <hr/>
  2363. <div class="block">
  2364. <name>V1.0.6.9</name>
  2365. <div>
  2366. <div>1、下载窗口加入关闭按钮。</div>
  2367. </div>
  2368. </div>
  2369. <div class="block">
  2370. <name>V1.0.6.8</name>
  2371. <div>
  2372. <div>1、修复夸克网盘按钮错位。</div>
  2373. </div>
  2374. </div>
  2375. <div class="block">
  2376. <name>V1.0.6.7</name>
  2377. <div>
  2378. <div>1、将百度网盘界面修改为主题色,可在设置选择是否修改;</div>
  2379. <div>2、增加主题色名称,更改部分内容颜色;</div>
  2380. <div>3、移动云盘API 下载支持批量复制;</div>
  2381. <div>4、优化控制台输出结果;</div>
  2382. <div>5、百度网盘API 下载不使用IDM时可以显示剩余时间;</div>
  2383. <div>6、“取消点亮按钮”按钮的位置现已移动到设置页面。</div>
  2384. <div>7homo特有的彩蛋又回来力(喜)。</div>
  2385. </div>
  2386. </div>
  2387. <div class="block">
  2388. <name>V1.0.6.6</name>
  2389. <div>
  2390. <div>1、修复暗号错误。</div>
  2391. </div>
  2392. </div>
  2393. <div class="block">
  2394. <name>V1.0.6.5</name>
  2395. <div>
  2396. <div>1、修复即使输入正确暗号也不能成功点亮按钮的服务器错误。</div>
  2397. </div>
  2398. </div>
  2399. <div class="block">
  2400. <name>V1.0.6.4</name>
  2401. <div>
  2402. <div>1、跟进官方V6.1.1版本,修复阿里云盘获取下载链接时的问题。</div>
  2403. </div>
  2404. </div>
  2405. <div class="block">
  2406. <name>V1.0.6.3</name>
  2407. <div>
  2408. <div>1、照顾小屏幕用户,将始终显示复制全部链接的按钮;</div>
  2409. <div>2、增加取消下载时的动画。</div>
  2410. </div>
  2411. </div>
  2412. <div class="block">
  2413. <name>V1.0.6.2</name>
  2414. <div>
  2415. <div>1、修复部分界面错位,实现CSS内置;</div>
  2416. <div>2、百度网盘界面将变得更加简洁。</div>
  2417. </div>
  2418. </div>
  2419. <div class="block">
  2420. <name>V1.0.6.1</name>
  2421. <div>
  2422. <div>1、新增百度云盘API 下载支持复制链接;</div>
  2423. <div>2、为了照顾手机浏览器用户,增大项目之间间隙,新增隐藏IDM提示选项,可在助手设置中启用;</div>
  2424. <div>3、修改CSS,界面会出现更多的主题色;</div>
  2425. <div>4、支持在游小猴官网查看暗号;</div>
  2426. <div>5、修复部分语法错误。</div>
  2427. </div>
  2428. </div>
  2429. <div class="block">
  2430. <name>V1.0.6</name>
  2431. <div>
  2432. <div>1、修复了打开阿里云盘分享连接时因下载移动端广告导致只能点击 API 下载;</div>
  2433. <div>2、跟进官方6.0.4版本,修复夸克网盘获取下载链接失效、支持移动云盘。</div>
  2434. </div>
  2435. </div>
  2436. <hr/>
  2437. <div class="block">
  2438. <name>V1.0.5.5</name>
  2439. <div>
  2440. <div>1、感谢<a href="https://github.com/Night-stars-1">Night-stars-1</a>的帮助,修复因为原作者服务器导致的初始化暗号识别错误;</div>
  2441. <div>2、修改一些文本以及提供给服务器的信息。</div>
  2442. </div>
  2443. </div>
  2444. <div class="block">
  2445. <name>V1.0.5.4</name>
  2446. <div>
  2447. <div>1、小修小改css,让主题色出现在更多地方;</div>
  2448. <div>2、修改下载链接获取失败的提示;</div>
  2449. <div>3、增加更多的主题色,可在助手设置查看;</div>
  2450. <div>4homo彩蛋被删去力(悲)。</div>
  2451. </div>
  2452. </div>
  2453. <div class="block">
  2454. <name>V1.0.5.3</name>
  2455. <div>
  2456. <div>1、修啦修啦,阿里云盘可以摸到下载菜单了。</div>
  2457. </div>
  2458. </div>
  2459. <div class="block">
  2460. <name>V1.0.5.2</name>
  2461. <div>
  2462. <div>1、增加脚本信息菜单(没有用);</div>
  2463. <div>2、优化阿里云盘显示svg图片;</div>
  2464. <div>3、修改弹窗按钮颜色。</div>
  2465. </div>
  2466. </div>
  2467. <div class="block">
  2468. <name>V1.0.5.1</name>
  2469. <div>
  2470. <div>1、修复在切换按钮主题后夸克网盘不能正常显示按钮。</div>
  2471. </div>
  2472. </div>
  2473. <div class="block">
  2474. <name>V1.0.5</name>
  2475. <div>
  2476. <div>1、跟进官方V5.0.4版本;</div>
  2477. <div>2、小改动,照着官方版本更正文件名称检测;</div>
  2478. <div>3、保留彩蛋,但必须舍弃官方暗号。</div>
  2479. </div>
  2480. </div>
  2481. <div class="block">
  2482. <name>V1.0.4</name>
  2483. <div>
  2484. <div>大改!</div>
  2485. <div>1、修复了原作者留下的夸克网盘切换文件夹就多一个“下载助手”按钮的大BUG;</div>
  2486. <div>2、终于来了,在下载菜单增加“助手设置”“更新日志”按钮;</div>
  2487. <div>【再也不用点进油猴管理再进设置了(保留油猴管理内设置)】</div>
  2488. <div>3、修改阿里云盘和夸克网盘下载助手按钮样式;</div>
  2489. <div>4、增加“取消点亮按钮”油猴菜单;</div>
  2490. <div>5、修改部分css,使其与选择的主题更贴切。</div>
  2491. </div>
  2492. </div>
  2493. <hr/>
  2494. <div class="block">
  2495. <name>V1.0.3</name>
  2496. <div>
  2497. <div>1、增加一个小彩蛋; 提示:</div>
  2498. <div>homo(需在未点亮按钮状态触发)</div>
  2499. <div>【需要重新恢复按钮为未点亮状态请进入 已安装脚本->编辑->开发者->重置到出厂->确定】</div>
  2500. <div>2、修改/增加默认主题色。</div>
  2501. </div>
  2502. </div>
  2503. <div class="block">
  2504. <name>V1.0.2</name>
  2505. <div>
  2506. <div>1、修改并加宽界面,调整部分css,使Sweetalert2界面更美观,更与原版相近;</div>
  2507. <div>2、修改部分提示文字,使文字更容易复制。</div>
  2508. </div>
  2509. </div>
  2510. <div class="block">
  2511. <name>V1.0.1</name>
  2512. <div>
  2513. <div>1、去除更新提示;</div>
  2514. <div>2、更新Sweetalert211版本;</div>
  2515. <div>3、部分CDN节点更换为jsdelivr。</div>
  2516. </div>
  2517. </div>
  2518. <div class="block">
  2519. <name>V1.0.0</name>
  2520. <div>
  2521. <div>1、增加“注入”功能(bushi);</div>
  2522. <div>2、去除广告。</div>
  2523. </div>
  2524. </div>
  2525. </div>
  2526. <style>
  2527. div:where(.swal2-container) div:where(.swal2-popup){
  2528. width:36em!important;
  2529. }
  2530. .version-log{
  2531. text-align:left;
  2532. }
  2533. .version-log > .block,
  2534. .version-log > hr{
  2535. margin-bottom:20px;
  2536. }
  2537. .version-log > hr{
  2538. border-style:inset;
  2539. border-width:1px;
  2540. }
  2541. .version-log .block name{
  2542. display:block;
  2543. margin-bottom:10px;
  2544. font-size:1.2em;
  2545. }
  2546. .version-log .block div{
  2547. margin-bottom:5px;
  2548. }
  2549. .version-log .block blockquote{
  2550. padding:0.7em;
  2551. border-left:5px solid #bdbdbd;
  2552. background-color:#f9f9f9;
  2553. margin:0;
  2554. }
  2555. @media (prefers-color-scheme:dark){
  2556. .version-log .block blockquote{
  2557. border-left:5px solid #7A7C84;
  2558. background-color:#464851;
  2559. }
  2560. }
  2561. </style>`,
  2562. allowOutsideClick: false,
  2563. showCloseButton: true,
  2564. confirmButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"/></svg> 我已阅',
  2565. });
  2566. },
  2567. /**
  2568. * 创建浮动提示框
  2569. * @description 监听元素悬停事件,动态显示带文件大小的提示框
  2570. * @author 油小猴
  2571. * @author hmjz100
  2572. * @fires .listener-tip - 鼠标移动事件触发提示框定位
  2573. * @see {@link temp.color} 使用全局主题色渲染文件大小信息
  2574. */
  2575. createTip() {
  2576. let tooltip = $('<div class="pl-tooltip"></div>');
  2577. this.waitForKeyElements(`.${mount}`, (element) => {
  2578. if ($('.pl-tooltip').length <= 0) element.append(tooltip);
  2579. return true;
  2580. }, true);
  2581. // 提取公共显示逻辑
  2582. const showTooltip = (target, x, y) => {
  2583. if (!target.data("title") && !target.data("size") && !target.text()) return;
  2584. let tip = "";
  2585. let title = target.data("title");
  2586. if (title) {
  2587. tip += `<span>${title}</span>`;
  2588. } else {
  2589. let name = target.find(".name").text();
  2590. let size = target.find(".size").text();
  2591. tip += `<span>${name}</span>`;
  2592. if (size !== undefined) {
  2593. tip += `<span style="background-color:${temp.color}">${size}</span>`;
  2594. }
  2595. }
  2596. tooltip.html(tip).css({
  2597. left: x + 10 + 'px',
  2598. top: y + 20 + 'px',
  2599. display: 'flex'
  2600. });
  2601. };
  2602. // 鼠标/触摸移动时更新位置
  2603. $doc.on('mousemove touchmove', '.listener-tip', function (e) {
  2604. const isTouch = e.type.startsWith('touch');
  2605. const pageX = isTouch ? e.originalEvent.touches[0].pageX : e.pageX;
  2606. const pageY = isTouch ? e.originalEvent.touches[0].pageY : e.pageY;
  2607. showTooltip($(e.currentTarget), pageX, pageY);
  2608. });
  2609. // 触摸开始时立即显示
  2610. $doc.on('touchstart', '.listener-tip', function (e) {
  2611. // 阻止滚动避免干扰
  2612. e.preventDefault();
  2613. const touch = e.originalEvent.touches[0];
  2614. showTooltip($(e.currentTarget), touch.pageX, touch.pageY);
  2615. });
  2616. // 统一隐藏逻辑
  2617. $doc.on('mouseleave touchend touchcancel', '.listener-tip, .pl-tooltip', function (e) {
  2618. tooltip.css({ display: "" });
  2619. });
  2620. // 增强跨事件类型检测
  2621. $doc.on('mousemove touchmove', function (e) {
  2622. const $tooltip = tooltip;
  2623. if (!$tooltip.is(':visible')) return;
  2624. const isTouch = e.type.startsWith('touch');
  2625. const touch = isTouch ? e.originalEvent.touches[0] : null;
  2626. const target = isTouch
  2627. ? document.elementFromPoint(touch.clientX, touch.clientY)
  2628. : e.target;
  2629. if (!$(target).closest('.listener-tip, .pl-tooltip').length) {
  2630. $tooltip.css({ display: "" });
  2631. }
  2632. });
  2633. },
  2634. /**
  2635. * 创建加载状态弹窗
  2636. * @author 油小猴
  2637. * @description 生成包含旋转动画的加载容器
  2638. */
  2639. createLoading() {
  2640. return $('<div class="pl-loading"><div class="pl-loading-box"><div><div></div><div></div></div></div></div>');
  2641. },
  2642. /**
  2643. * 创建用于下载的隐藏 iframe
  2644. * @author 油小猴
  2645. * @description 该方法会创建一个隐藏的 iframe 元素,并将其插入到指定的挂载点中,用于后续的下载操作。
  2646. * iframe 的 src 设置为 "javascript:;" 以避免加载额外资源,提升性能。
  2647. */
  2648. createDownloadIframe() {
  2649. let iframe = $('<iframe style="padding:0;margin:0;display:block;display:none" src="javascript:;" id="downloadIframe"></iframe>');
  2650. base.waitForKeyElements(`.${mount}`, (element) => {
  2651. element.append(iframe);
  2652. }, true)
  2653. },
  2654. /**
  2655. * 创建用于下载页面的 HTML
  2656. * @author 油小猴
  2657. * @author hmjz100
  2658. * @param {Array} configs - 用于配置生成 HTML 的参数
  2659. * @returns {string} 生成的 HTML 内容
  2660. * @description 详见代码
  2661. */
  2662. generateDom(configs) {
  2663. if (base.isType(configs) !== "array" && configs.length !== 2) return message.error('提示:<br/>配置解析失败~');
  2664. let list = (Array.isArray(configs[0]) ? configs[0] : []);
  2665. if (!list.length) return message.error('提示:<br/>获取下载链接失败,刷新网页后再试试吧~');
  2666. let {
  2667. isFolder,
  2668. getFileName,
  2669. getFileSize,
  2670. getFileLink,
  2671. getFileMirror,
  2672. convert = {},
  2673. tooltip = {}
  2674. } = (base.isType(configs[1]) === 'object' ? configs[1] : {});
  2675. let content = $(`<div><div class="pl-main"></div><div class="pl-extra"></div></div>`);
  2676. let allLink = [];
  2677. list.forEach((v, i) => {
  2678. if (isFolder(v)) return;
  2679. let filename = getFileName(v);
  2680. let size = getFileSize(v);
  2681. let dlink = getFileLink(v);
  2682. let mirror = base.isType(getFileMirror) !== "undefined" ? getFileMirror(getFileLink(v)) : undefined;
  2683. if (!dlink || !dlink.includes("http")) {
  2684. content.find(".pl-main").append(`<div class="pl-item">
  2685. <div class="pl-item-name listener-tip" data-size="${size}"><div class="name">${filename}</div><div class="size">${base.sizeFormat(size)}</div></div>
  2686. <div class="pl-item-tip">获取下载链接失败,刷新网页后再试试吧~</div>
  2687. </div>`)
  2688. } else {
  2689. if (temp.mode === "api") {
  2690. allLink.push(dlink);
  2691. content.find(".pl-main").append(`<div class="pl-item">
  2692. <div class="pl-item-name listener-tip"><div class="name">${filename}</div><div class="size">${base.sizeFormat(size)}</div></div>
  2693. <button class="pl-item-link pl-btn-primary pl-btn-default listener-api-download enhance listener-tip" data-index="${i}" data-link="${dlink}" data-filename="${filename}" data-size="${size}" data-title="建议换用 “Aria2 推送下载”,本方式若文件过大(具体取决于浏览器限制)可能会导致无法下载<br/>通过脚本跨域请求下载文件,适用于较新的浏览器,可在此显示剩余时间和速度"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>增强下载 (基于脚本跨域)</button>
  2694. <button class="pl-item-link pl-btn-primary pl-btn-info listener-api-download normal listener-tip" data-link="${dlink}" data-filename="${filename}" data-title="通过浏览器访问链接下载文件,适用于支持 iframe 的浏览器<br/>点击后需等待浏览器弹出提示才可点击下个下载,否则只会下载后者"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>直接下载 (基于访问链接)</button>
  2695. <button class="pl-item-copy pl-btn-primary pl-btn-success listener-copy listener-tip" data-copy='${filename}' data-title="点击复制文件名"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制名称</button>
  2696. <button class="pl-item-copy pl-btn-primary pl-btn-warning listener-copy copy listener-tip" data-copy='${dlink}' data-title="点击复制下载链接"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制链接</button>
  2697. <div class="pl-item-progress" style="display:none">
  2698. <div class="pl-progress">
  2699. <div class="pl-progress-outer"></div>
  2700. <div class="pl-progress-inner" style="width:5%">
  2701. <div class="pl-progress-inner-text">正在加载进度...0%</div>
  2702. </div>
  2703. </div>
  2704. <button class="pl-btn-primary pl-btn-danger pl-progress-stop listener-stop"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg>取消下载</button>
  2705. <button class="pl-btn-primary pl-btn-info pl-progress-back listener-back" style="display:none"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg>返回</button>
  2706. </div>
  2707. </div>`);
  2708. }
  2709. if (temp.mode === "curl") {
  2710. let finalink = base.convertLinkToCurl(dlink, filename, convert?.curl);
  2711. allLink.push(finalink);
  2712. content.find(".pl-main").append(`<div class="pl-item">
  2713. <div class="pl-item-name listener-tip" data-size="${size}"><div class="name">${filename}</div><div class="size">${base.sizeFormat(size)}</div></div>
  2714. <a class="pl-item-link listener-copy listener-tip" data-copy='${finalink}' data-title="点击复制 curl 命令行">${finalink}<br/><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制 ${filename} 下载命令行</a>
  2715. </div>`);
  2716. }
  2717. if (temp.mode === "aria2") {
  2718. let finalink = base.convertLinkToAria2(dlink, filename, convert?.aria2);
  2719. allLink.push(finalink);
  2720. content.find(".pl-main").append(`<div class="pl-item">
  2721. <div class="pl-item-name listener-tip" data-size="${size}"><div class="name">${filename}</div><div class="size">${base.sizeFormat(size)}</div></div>
  2722. <button class="pl-item-link pl-btn-primary pl-btn-default listener-aria2-download" data-filename="${filename}" data-link="${dlink}"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-up"/></svg><span>推送链接到 Aria2 下载器</span></button>
  2723. <button class="pl-btn-primary pl-btn-info listener-copy listener-tip" data-copy='${finalink}' data-title="Aria2 没启用 RPC?点击复制 aria2c 命令行手动下载"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制下载命令行</button>
  2724. </div>`);
  2725. }
  2726. if (temp.mode === "bitcomet") {
  2727. let finalink = base.convertLinkToBitComet(dlink, filename, convert?.bitcomet);
  2728. allLink.push(finalink);
  2729. content.find(".pl-main").append(`<div class="pl-item">
  2730. <div class="pl-item-name listener-tip" data-size="${size}"><div class="name">${filename}</div><div class="size">${base.sizeFormat(size)}</div></div>
  2731. <a class="pl-item-link pl-btn-primary pl-btn-default listener-tip" href="${finalink}" data-title="点击打开 BC 链接以手动调起比特彗星下载,右键可复制 BC 链接"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>使用 BC 链接下载</a>
  2732. ${mirror ? `<button class="pl-btn-primary pl-btn-success listener-copy listener-tip" data-copy='${mirror}' data-title="点击复制镜像地址"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制镜像</a>` : ""}
  2733. <button class="pl-btn-primary pl-btn-info listener-bitcomet-download listener-tip" data-filename="${filename}" data-link="${dlink}" data-title="除非 BC 链接无法调起比特彗星,否则不建议使用此方式<br/><br/>由于比特彗星内置的远程下载 Web API 服务代码存在缺陷,请求可能会随机出现“发送失败 - 服务器返回空请求”错误,实际上客户端已成功开始下载<br/>由于脚本无法准确判断请求是否真正成功,即使出现错误,也会提示“成功”"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-up"/></svg><span>推送到下载器</span></button>
  2734. </div>`);
  2735. }
  2736. if (temp.mode === "abdm") {
  2737. content.find(".pl-main").append(`<div class="pl-item">
  2738. <div class="pl-item-name listener-tip" data-size="${size}"><div class="name">${filename}</div><div class="size">${base.sizeFormat(size)}</div></div>
  2739. <button class="pl-item-link pl-btn-primary pl-btn-default listener-abdm-download slient" data-filename="${filename}" data-link="${dlink}"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-up"/></svg><span>推送链接到 ABDM 下载器</span></button>
  2740. </div>`);
  2741. }
  2742. }
  2743. });
  2744. allLink = (allLink ? allLink.join("\r\n") : "")
  2745. if (temp.mode === "api" && list.length >= 2) {
  2746. content.find(".pl-extra").append(`<button class="pl-btn-primary api listener-download-all enhance listener-tip" data-title="建议换用 “Aria2 推送下载”,本方式若文件过大有可能不会弹出下载后窗口<br/>通过脚本跨域请求下载文件,适用于较新的浏览器,可在此显示剩余时间和速度"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>全部增强下载</button>
  2747. <button class="pl-btn-primary pl-btn-warning api listener-copy listener-tip" data-copy="${allLink}" data-title="点击复制全部下载链接"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制全部链接</button>`);
  2748. } else if (temp.mode === "curl") {
  2749. content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-warning curl listener-open-setting listener-tip" data-title="${temp.terminalType[base.getValue('setting_curl_terminal')]}" data-back-to-downloads="true"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>修改终端类型</button>`);
  2750. if (list.length >= 2) content.find(".pl-extra").append(`<button class="pl-btn-primary curl listener-copy listener-tip" data-copy='${allLink}' data-title="点击复制全部 curl 命令行"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制全部命令行</button>`);
  2751. } else if (temp.mode === 'aria2') {
  2752. let rpc = base.getValue("setting_aria2_rpc").find(i => i.default);
  2753. content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-warning aria2 listener-open-aria2-setting listener-tip" data-title="${rpc.domain + ':' + rpc.port + rpc.path}" data-back-to-downloads="true"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>修改服务参数</button>`);
  2754. content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-success aria2 listener-rpc-task youxiaohou listener-tip" data-title="访问原作者的 Aria2 管理页面以查看下载任务,功能较少"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-list-check"/></svg>查看任务 (油小猴)</button>`);
  2755. content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-success aria2 listener-rpc-task ariang listener-tip" data-title="访问 AriaNg 的官方 Demo 以查看下载任务,功能较多"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-list-check"/></svg>查看任务 (AriaNg)</button>`);
  2756. if (list.length >= 2) content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-default aria2 listener-send-rpc" data-type="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-up"/></svg>推送全部到下载器</button>`);
  2757. if (list.length >= 2) content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-info aria2 listener-copy listener-tip" data-copy='${allLink}' data-title="Aria2 没启用 RPC?点击复制 aria2c 命令行手动下载"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制全部命令行</button>`);
  2758. } else if (temp.mode === "bitcomet") {
  2759. let rpc = base.getValue("setting_bitcomet_rpc").find(i => i.default);
  2760. content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-warning bitcomet listener-open-bitcomet-setting listener-tip" data-title="${rpc.domain + ':' + rpc.port + rpc.path}" data-back-to-downloads="true"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>修改服务参数</button>`);
  2761. if (list.length >= 2) content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-default bitcomet listener-copy listener-tip" data-copy='${allLink}' data-title="点击复制全部 BC 链接"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-copy"/></svg>复制全部 BC 链接</button>`);
  2762. if (list.length >= 2) content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-info bitcomet listener-send-rpc listener-tip" data-type="bitcomet" data-title="除非 BC 链接无法调起比特彗星,否则不建议使用此方式<br/><br/>由于比特彗星内置的远程下载 Web API 服务代码存在缺陷,请求可能会随机出现“发送失败 - 服务器返回空请求”错误,实际上客户端已成功开始下载<br/>由于脚本无法准确判断请求是否真正成功,即使出现错误,也会提示“成功”"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-up"/></svg>推送全部到下载器</button>`);
  2763. } else if (temp.mode === 'abdm') {
  2764. let rpc = base.getValue("setting_abdm_rpc").find(i => i.default);
  2765. content.find(".pl-extra").append(`<button class="pl-btn-primary pl-btn-warning abdm listener-open-abdm-setting listener-tip" data-title="${rpc.domain + ':' + rpc.port}" data-back-to-downloads="true"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>修改服务参数</button>`);
  2766. }
  2767. function updateTooltip($element, value) {
  2768. if (!value) return;
  2769. $element.addClass("listener-tip");
  2770. if (value.startsWith('+')) {
  2771. // 追加模式:去掉开头的 '+',然后拼接到现有 data-title
  2772. const newValue = value.substring(1);
  2773. const existingTitle = $element.attr("data-title") || "";
  2774. $element.attr("data-title", existingTitle + newValue);
  2775. } else {
  2776. // 替换模式
  2777. $element.attr("data-title", value);
  2778. }
  2779. }
  2780. if (tooltip?.enhance) updateTooltip(content.find(".enhance"), tooltip.enhance);
  2781. if (tooltip?.normal) updateTooltip(content.find(".normal"), tooltip.normal);
  2782. if (tooltip?.copy) updateTooltip(content.find(".copy"), tooltip.copy);
  2783. if (tooltip?.filename) updateTooltip(content.find(".filename"), tooltip.filename);
  2784. let html = content.html();
  2785. content.remove();
  2786. return html;
  2787. },
  2788. /**
  2789. * 获取镜像列表
  2790. * @author 油小猴
  2791. * @description 根据原始链接和镜像域名列表生成多个镜像链接,支持多线程下载。
  2792. * 每个镜像地址会根据 thread 参数生成多个重复链接(通过添加 `&` 符号区分)。
  2793. * @param {string} link - 原始下载链接
  2794. * @param {Array<string>} mirror - 镜像域名数组
  2795. * @param {number} [thread=2] - 每个镜像生成的线程数(链接重复次数),默认为 2
  2796. * @returns {string} 所有镜像链接组成的字符串,每行一个链接
  2797. *
  2798. * @example
  2799. * getMirrorList("https://example.com/file.zip", ["mirror1.com", "mirror2.com"], 2)
  2800. * // 返回:
  2801. * // https://mirror1.com/file.zip
  2802. * // https://mirror1.com/file.zip&
  2803. * // https://mirror2.com/file.zip
  2804. * // https://mirror2.com/file.zip&
  2805. */
  2806. getMirrorList(link, mirror, thread = 2) {
  2807. try {
  2808. let host = new URL(link).host;
  2809. let mirrors = [];
  2810. for (let i = 0; i < mirror.length; i++) {
  2811. for (let j = 0; j < thread; j++) {
  2812. let item = link.replace(host, mirror[i]) + '&'.repeat(j);
  2813. mirrors.push(item);
  2814. }
  2815. }
  2816. return mirrors.join('\n');
  2817. } catch {
  2818. return undefined
  2819. }
  2820. },
  2821. /**
  2822. * 添加页面元素监听
  2823. * @author 油小猴
  2824. * @author hmjz100
  2825. * @description 详见代码
  2826. */
  2827. addPageListener() {
  2828. $doc.on('click', '.listener-open-setting', (e) => {
  2829. base.showSetting(e);
  2830. });
  2831. $doc.on('click', '.listener-open-aria2-setting', (e) => {
  2832. base.showAria2Setting(e);
  2833. });
  2834. $doc.on('click', '.listener-open-bitcomet-setting', (e) => {
  2835. base.showBitcometSetting(e);
  2836. });
  2837. $doc.on('click', '.listener-open-abdm-setting', (e) => {
  2838. base.showABDMSetting(e);
  2839. });
  2840. $doc.on('click', '.listener-open-updatelog', () => {
  2841. base.showUpdate();
  2842. });
  2843. $doc.on('click', '.listener-open-beautify', () => {
  2844. base.showBeautify();
  2845. });
  2846. $doc.on('click', '.listener-unregister', async function (e) {
  2847. message.warning("正在“注入”设置项目...");
  2848. let list = base.getValue("setting_init");
  2849. list.code = "";
  2850. list.license = "";
  2851. base.setValue('setting_init', list);
  2852. base.delValue('baidu_access_token');
  2853. location.reload();
  2854. });
  2855. $doc.on('change', '.listener-terminal', async function (e) {
  2856. base.setValue('setting_curl_terminal', e.currentTarget.value);
  2857. });
  2858. $doc.on('click', '.listener-color', async function (e) {
  2859. let element = $(e.currentTarget).closest('.listener-color').length > 0 ? $(e.currentTarget).closest('.listener-color') : $(e.currentTarget);
  2860. let parent = element.closest('.pl-color');
  2861. let mask = element.find(".mask");
  2862. let color = element.data('color');
  2863. if (color && parent.length > 0 && mask.length > 0) {
  2864. parent.find(".this").remove();
  2865. mask.append(`<div class="this"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"></use></svg></div>`);
  2866. let list = base.getValue("setting_ui_theme")
  2867. list.color = color;
  2868. base.setValue("setting_ui_theme", list);
  2869. base.addPanLinkerStyle();
  2870. }
  2871. });
  2872. $doc.on('change', '.listener-theme', async function (e) {
  2873. let list = base.getValue("setting_ui_theme");
  2874. list.custom[e.currentTarget.dataset.type] = e.currentTarget.checked;
  2875. base.setValue("setting_ui_theme", list);
  2876. });
  2877. $doc.on('click', '.listener-api-download.normal', async function (e) {
  2878. e.preventDefault();
  2879. let dataset = e.currentTarget.dataset;
  2880. let link = new URL(dataset.link);
  2881. $('#downloadIframe').attr('src', link.href);
  2882. });
  2883. $doc.on('click', '.listener-retry', async function (e) {
  2884. let o = base._EventFactory(e);
  2885. o.tip.hide();
  2886. o.link.show();
  2887. o.directLink.show();
  2888. });
  2889. $doc.on('click', '.listener-stop', async function (e) {
  2890. let o = base._EventFactory(e);
  2891. let index = o.link[0].dataset.index;
  2892. if (temp.request[index]) {
  2893. temp.request[index].abort();
  2894. clearInterval(temp.ins[index]);
  2895. o.item.find('.pl-progress-inner-text').text('正在取消...');
  2896. o.item.find('.pl-progress-inner').css('width', 100 + '%');
  2897. await base.sleep(1050);
  2898. o.tip.hide();
  2899. o.back.hide();
  2900. o.link.show();
  2901. o.directLink.show();
  2902. o.copy.show();
  2903. o.progress.hide();
  2904. o.stop.hide();
  2905. }
  2906. });
  2907. $doc.on('click', '.listener-back', async function (e) {
  2908. let o = base._EventFactory(e);
  2909. o.progress.hide();
  2910. o.tip.hide();
  2911. o.link.show();
  2912. o.directLink.show();
  2913. o.copy.show();
  2914. o.stop.hide();
  2915. o.back.hide();
  2916. });
  2917. $doc.on('click', '.listener-download-all', async function (e) {
  2918. let target = $(e.currentTarget);
  2919. let originalHtml = target.html();
  2920. $('.pl-item-link.enhance').each((index, element) => {
  2921. if ($(element).css('display') !== 'none') {
  2922. $(element).click();
  2923. }
  2924. });
  2925. target.text('下载开始,进度见上方按钮哦~').animate({ opacity: '0.5' }, "slow");
  2926. await base.sleep(2000);
  2927. target.css('opacity', "");
  2928. target.html(originalHtml);
  2929. });
  2930. $doc.on('click', '.listener-send-rpc', async function (e) {
  2931. let target = $(e.currentTarget);
  2932. let originalHtml = target.html();
  2933. $(`.listener-${target.data("type")}-download`).each((index, element) => {
  2934. if ($(element).attr('data-processing') !== 'true') {
  2935. $(element).click();
  2936. }
  2937. });
  2938. target.text('发送完成,结果见上方按钮哦~').animate({ opacity: '0.5' }, "slow");
  2939. await base.sleep(2000);
  2940. target.css('opacity', "");
  2941. target.html(originalHtml);
  2942. });
  2943. $doc.on('click', '.listener-copy', async function (e) {
  2944. e.preventDefault();
  2945. let target = $(e.currentTarget);
  2946. let originalHtml = target.html();
  2947. let copy = target.data("copy");
  2948. if (copy) {
  2949. base.setClipboard(copy)
  2950. target.html(`<svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"/></svg>复制成功`).animate({ opacity: '0.5' }, "slow");
  2951. await base.sleep(2000);
  2952. target.css('opacity', "");
  2953. target.html(originalHtml);
  2954. }
  2955. });
  2956. $doc.on('click', '.listener-rpc-task.youxiaohou', function () {
  2957. let rpc = base.getValue("setting_aria2_rpc").find(i => i.default);
  2958. let isHttps = rpc.domain.startsWith('https://');
  2959. let url = `${isHttps ? 'https' : 'http'}://d.youxiaohou.com/?rpc=${base.encodeBase(JSON.stringify({ domain: rpc.domain, port: rpc.port }))}#${rpc.token}`;
  2960. GM_openInTab(url, { active: true, insert: true, setParent: true });
  2961. });
  2962. $doc.on('click', '.listener-rpc-task.ariang', function () {
  2963. let rpc = base.getValue("setting_aria2_rpc").find(i => i.default);
  2964. let isHttps = rpc.domain.startsWith('https://');
  2965. let url = `${isHttps ? 'https' : 'http'}://ariang.mayswind.net/latest/#!/settings/rpc/set?protocol=${isHttps ? 'wss' : 'ws'}&host=${rpc.domain.replace(/^(https?:\/\/)/, "")}&port=${rpc.port}&interface=${rpc.path.replace(/^\//, "")}&secret=${rpc.token}`;
  2966. GM_openInTab(url, { active: true, insert: true, setParent: true });
  2967. });
  2968. $doc.on('change', '.listener-rpc-select', async function (e) {
  2969. let element = $(this);
  2970. let selectedIndex = element.val();
  2971. let type = element.data("type");
  2972. let list = base.getValue(`setting_${type}_rpc`);
  2973. if (selectedIndex === 'new') {
  2974. return $('.listener-rpc-input').val("");
  2975. } else if (list[selectedIndex]) {
  2976. list.forEach((item, index) => {
  2977. if (item.default) {
  2978. delete item.default;
  2979. }
  2980. });
  2981. list[selectedIndex].default = true;
  2982. base.setValue(`setting_${type}_rpc`, list);
  2983. $('.listener-rpc-input').each((index, element) => {
  2984. let type = $(element).data('type').split(".")[1];
  2985. $(element).val(list[selectedIndex][type] || "");
  2986. });
  2987. }
  2988. });
  2989. $doc.on('input', '.listener-rpc-input', async function (e) {
  2990. let type = $(this).data("type");
  2991. if (!type) return;
  2992. type = type.split(".")
  2993. let list = base.getValue(`setting_${type[0]}_rpc`);
  2994. let value = $(this).val();
  2995. let selectedIndex = $('.listener-rpc-select option:selected').val();
  2996. if (selectedIndex === 'new') {
  2997. list.forEach((item, index) => {
  2998. if (item.default) {
  2999. delete item.default;
  3000. }
  3001. });
  3002. list.push({
  3003. domain: "",
  3004. port: "",
  3005. path: "",
  3006. default: true
  3007. });
  3008. base.setValue(`setting_${type[0]}_rpc`, list);
  3009. selectedIndex = list.length - 1
  3010. }
  3011. list[selectedIndex][type[1]] = value;
  3012. base.setValue(`setting_${type[0]}_rpc`, list)
  3013. let select = $('.listener-rpc-select');
  3014. let options = list.map((item, index) => {
  3015. return `<option value="${index}"${item.default ? ' selected' : ""}>${item.domain ? item.domain : ""}:${item.port ? item.port : ""}${item.path ? item.path : ""}</option>`;
  3016. }).join("");
  3017. select.html(`${options}<option value="new">+ 创建新项目</option>`);
  3018. });
  3019. $doc.on('click', '.listener-rpc-delete', async function (e) {
  3020. let type = $(this).data("type");
  3021. let list = base.getValue(`setting_${type}_rpc`);
  3022. let selectedIndex = parseInt($('.listener-rpc-select option:selected').val(), 10);
  3023. if (selectedIndex === 'new' || !confirm('您确定要删除此项目吗?')) return;
  3024. list = list.filter((_, i) => i !== selectedIndex);
  3025. if (list.length === 0) return alert('至少保留一个配置');
  3026. let newDefaultIndex = selectedIndex === 0 ? 0 : selectedIndex - 1;
  3027. list[newDefaultIndex].default = true;
  3028. base.setValue(`setting_${type}_rpc`, list);
  3029. let select = $('.listener-rpc-select');
  3030. let options = list.map((item, index) => {
  3031. return `<option value="${index}"${item.default ? ' selected' : ""}>${item.domain ? item.domain : ""}:${item.port ? item.port : ""}${item.path ? item.path : ""}</option>`;
  3032. }).join("");
  3033. select.html(`${options}<option value="new">+ 创建新项目</option>`);
  3034. $('.listener-rpc-input').each(function () {
  3035. let key = $(this).data('type').split(".")[1];
  3036. $(this).val(list[newDefaultIndex][key] || "");
  3037. });
  3038. });
  3039. $doc.on('click', '.listener-rpc-test', async function (e) {
  3040. let element = $(this);
  3041. let type = element.data("type");
  3042. let selectedIndex = $('.listener-rpc-select option:selected').val();
  3043. let list = base.getValue(`setting_${type}_rpc`);
  3044. let text = element.find("span");
  3045. let originalHtml = text.html();
  3046. if (selectedIndex === 'new' || element.data("testing") === "true") return;
  3047. if (list[selectedIndex]) {
  3048. element.data("testing", "true");
  3049. text.html("等待");
  3050. element.css({ 'opacity': '0.9' });
  3051. let selected = list.find(i => i.default);
  3052. let result = "fail"
  3053. if (type === "aria2") {
  3054. let domain = selected.domain,
  3055. port = selected.port,
  3056. path = selected.path,
  3057. token = selected.token;
  3058. result = await base.testConnectToAria2(domain, port, path, token);
  3059. } else if (type === "abdm") {
  3060. let domain = selected.domain,
  3061. port = selected.port;
  3062. result = await base.testConnectToABDM(domain, port);
  3063. }
  3064. if (result === "success") {
  3065. text.html("成功");
  3066. element.css({ 'background-color': '#52c41a' });
  3067. } else {
  3068. text.html("失败");
  3069. element.css({ 'background-color': '#cb1616' });
  3070. }
  3071. element.css({ 'opacity': "" });
  3072. await base.sleep(3000);
  3073. element.data("testing", "false");
  3074. text.html(originalHtml);
  3075. element.css({ 'background-color': "" });
  3076. }
  3077. });
  3078. },
  3079. /**
  3080. * 创建基础样式
  3081. * @author 油小猴
  3082. * @author hmjz100
  3083. * @description 为组件添加默认的公共样式,包括浅色和深色模式适配样式。
  3084. */
  3085. addPanLinkerStyle() {
  3086. temp.color = base.getValue('setting_ui_theme').color;
  3087. base.addStyle('swal-pub-style', 'style', `@media (prefers-color-scheme:light){${GM_getResourceText('SwalLigt')}}`);
  3088. base.addStyle('swal-pub-dark-style', 'style', `@media (prefers-color-scheme:dark){${GM_getResourceText('SwalDark').replace(/#19191a/, '#222226')}}`);
  3089. base.addStyle('swal-pub-custom-style', 'style', `
  3090. .swal2-container *{vertical-align:baseline}
  3091. .swal2-styled{transition:all.2s}
  3092. .swal2-loader{display:none;align-items:center;justify-content:center;width:2.2em;height:2.2em;margin:0 1.875em;-webkit-animation:swal2-rotate-loading 1.5s linear 0s infinite normal;animation:swal2-rotate-loading 1.5s linear 0s infinite normal;border-width:.25em;border-style:solid;border-radius:100%;border-color:${temp.color} transparent }
  3093. .swal2-timer-progress-bar-container{position:absolute;right:0;bottom:0;left:0;grid-column:auto;overflow:hidden;border-bottom-right-radius:5px;border-bottom-left-radius:5px}
  3094. .swal2-timer-progress-bar{width:100%;height:.25em;background:${temp.color}33 }
  3095. .swal2-progress-steps .swal2-progress-step{z-index:20;flex-shrink:0;width:2em;height:2em;border-radius:2em;background:${temp.color};color:#fff;line-height:2em;text-align:center}
  3096. .swal2-progress-steps .swal2-progress-step.swal2-active-progress-step{background:${temp.color} }
  3097. .swal2-progress-steps .swal2-progress-step-line{z-index:10;flex-shrink:0;width:2.5em;height:.4em;margin:0 -1px;background:${temp.color}}
  3098. .swal2-html-container{padding:1em 1.6em 0.3em;margin:0}
  3099. .swal2-close,div:where(.swal2-container) button:where(.swal2-close){position:absolute;border-radius:10px;top:0;right:0;transition:all.2s}
  3100. .swal2-close:hover,div:where(.swal2-container) button:where(.swal2-close):hover{color:${temp.color};background-color:${temp.color}30;font-size:60px}
  3101. .swal2-styled{display:flex;justify-content:center;align-items:center;gap:5px}
  3102. .swal2-styled.swal2-confirm,div:where(.swal2-container) button:where(.swal2-styled):where(.swal2-confirm){background-color:${temp.color};color:#fff}
  3103. .swal2-styled.swal2-confirm:focus,div:where(.swal2-container) button:where(.swal2-styled):where(.swal2-confirm):focus{box-shadow:0 0 0 3px ${temp.color}80}
  3104. .swal2-styled.swal2-deny:focus,.swal2-close:focus,div:where(.swal2-container) button:where(.swal2-styled):where(.swal2-deny):focus{box-shadow:0 0 0 3px #dc374180}
  3105. .swal2-styled.swal2-cancel:focus,div:where(.swal2-container) button:where(.swal2-styled):where(.swal2-cancel):focus{box-shadow:0 0 0 3px #6e788180}
  3106. .swal2-styled.swal2-confirm,
  3107. .swal2-styled.swal2-deny,
  3108. .swal2-styled.swal2-cancel,
  3109. div:where(.swal2-container) button:where(.swal2-styled):where(.swal2-confirm),
  3110. div:where(.swal2-container) button:where(.swal2-styled):where(.swal2-deny),
  3111. div:where(.swal2-container) button:where(.swal2-styled):where(.swal2-cancel)
  3112. {border-radius:50px}
  3113. div:where(.swal2-container) div:where(.swal2-actions):not(.swal2-loading) .swal2-styled:hover{opacity:0.7}
  3114. .swal2-backdrop-show,.swal2-noanimation,div:where(.swal2-container).swal2-backdrop-show, div:where(.swal2-container).swal2-noanimation{background:rgba(25,25,26,.75);transition:backdrop-filter.2s;backdrop-filter:blur(1px)}
  3115. body.swal2-toast-shown .swal2-container{backdrop-filter:none}
  3116. .swal2-popup,div:where(.swal2-container) div:where(.swal2-popup){padding-bottom:1em;border-radius:10px}
  3117. .swal2-title,div:where(.swal2-container) h2:where(.swal2-title){height:auto}
  3118. .swal2-html-container,div:where(.swal2-container) div:where(.swal2-html-container){padding:1.3em 1.3em 0.3em;margin:0}
  3119. .swal2-footer,div:where(.swal2-container) div:where(.swal2-footer){flex-direction:column;justify-content:center;align-items:center}
  3120. .swal2-footer p,div:where(.swal2-container) div:where(.swal2-footer) p{margin:0;padding:0}
  3121. .swal2-icon-content,div:where(.swal2-icon) .swal2-icon-content{font-family:sans-serif}
  3122. .swal2-input, .swal2-file, swal2-select, .swal2-textarea,
  3123. div:where(.swal2-container) input:where(.swal2-input),
  3124. div:where(.swal2-container) input:where(.swal2-file),
  3125. div:where(.swal2-container) input:where(.swal2-select),
  3126. div:where(.swal2-container) textarea:where(.swal2-textarea)
  3127. {box-shadow:none}
  3128. .swal2-input:focus, .swal2-file:focus, .swal2-select:focus, .swal2-textarea:focus,
  3129. .swal2-input:focus-visible, .swal2-file:focus-visible, .swal2-select:focus-visible, .swal2-textarea:focus-visible,
  3130. div:where(.swal2-container) input:where(.swal2-input):focus,
  3131. div:where(.swal2-container) input:where(.swal2-input):focus-visible,
  3132. div:where(.swal2-container) input:where(.swal2-file):focus,
  3133. div:where(.swal2-container) input:where(.swal2-file):focus-visible,
  3134. div:where(.swal2-container) input:where(.swal2-select):focus,
  3135. div:where(.swal2-container) input:where(.swal2-select):focus-visible,
  3136. div:where(.swal2-container) textarea:where(.swal2-textarea):focus,
  3137. div:where(.swal2-container) textarea:where(.swal2-textarea):focus-visible
  3138. {outline:0;border:1px solid ${temp.color};box-shadow:0 0 0 3px ${temp.color}80}
  3139. .swal2-checkbox, .swal2-file, .swal2-input, .swal2-radio, .swal2-select, .swal2-textarea,
  3140. div:where(.swal2-container) input:where(.swal2-input), div:where(.swal2-container) input:where(.swal2-file), div:where(.swal2-container) textarea:where(.swal2-textarea), div:where(.swal2-container) select:where(.swal2-select), div:where(.swal2-container) div:where(.swal2-radio), div:where(.swal2-container) label:where(.swal2-checkbox)
  3141. {margin:1em 2em}`);
  3142. base.addStyle(`${mount}-main-style`, 'style', `
  3143. ::-webkit-scrollbar{width:8px;height:8px}
  3144. ::-webkit-scrollbar-track{border-radius:10px;background:#fff}
  3145. ::-webkit-scrollbar-thumb,::-webkit-scrollbar-thumb:hover{border-radius:10px}
  3146. ::-webkit-scrollbar-thumb{background-color:${temp.color}90!important,transition:background-color.2s;will-change:background-color}
  3147. ::-webkit-scrollbar-thumb:hover{background-color:${temp.color}D0!important}
  3148. .swal2-popup{font-size:16px}
  3149. .pl-popup{font-size:12px;min-width:70%;max-width:95%}
  3150. .pl-header{padding:0;align-items:flex-start;border-bottom:1px solid #eee;margin:0 0 10px;padding:0 0 5px}
  3151. .pl-title{font-size:18px;white-space:nowrap;text-overflow:ellipsis}
  3152. .pl-content{padding:0;font-size:12px}
  3153. .pl-footer{font-size:15px;text-align:center;display:block}
  3154. .pl-icon{width:15px;height:15px;vertical-align:-0.15em;fill:currentColor;overflow:hidden;font-size:18px}
  3155. .pl-main{background:${temp.color}15;border-radius:10px;display:flex;flex-direction:column;gap:8px;max-height:calc(${document.documentElement.clientHeight}px - 300px);overflow:auto;padding:8px 6px}
  3156. .pl-a{position:relative;vertical-align:baseline;color:${temp.color};border-bottom:2px solid ${temp.color};text-decoration:none!important;transition:color.3s,opacity.3s;will-change:color,opacity;overflow:hidden}
  3157. .pl-a::before{content:"";position:absolute;left:0;bottom:0;width:100%;height:100%;background-color:${temp.color};transform:scaleY(0);transform-origin:bottom center;transition:transform.15s,opacity.3s;will-change:transform;z-index:-1}
  3158. .pl-a:hover,.pl-a:focus{color:#fff}
  3159. .pl-a:hover::before,.pl-a:focus::before{transform:scaleY(1)}
  3160. .pl-a:active{color:#fff;opacity:0.8}
  3161. .pl-a .pl-icon{vertical-align:-0.06em;}
  3162. .pl-item{display:flex;align-items:center;background:${temp.color}30;border-radius:8px;padding:5px;gap:10px}
  3163. .pl-item-name{width:15%;text-align:left;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;cursor:default}
  3164. .pl-item-name>*{text-align:left;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}
  3165. .pl-item-link{flex:1;cursor:pointer}
  3166. a.pl-item-link{color:${temp.color};text-align:left;white-space:nowrap;text-overflow:ellipsis;overflow:hidden;transition:color.15s;will-change:color}
  3167. a.pl-item-link:hover{color:#fff}
  3168. .pl-item-tip{display:flex;justify-content:space-between;flex:1}
  3169. .pl-item-progress{display:flex;flex:1;align-items:center}
  3170. .pl-progress{display:inline-block;vertical-align:middle;width:100%;box-sizing:border-box;line-height:1;position:relative;height:20px;margin:0 6px;flex:1}
  3171. .pl-progress-outer{height:20px;border-radius:100px;background-color:#c1c1c1;overflow:hidden;position:relative;vertical-align:middle}
  3172. .pl-progress-inner{position:absolute;left:0;top:0;background-color:${temp.color};border-radius:100px;line-height:1;white-space:nowrap;transition:width.6s;will-change:width;height:20px;display:inline-flex;text-align:center;align-items:center}
  3173. .pl-progress-inner-text{display:inline-block;vertical-align:middle;cursor:default;color:#ffffff;font-size:12px;margin:0 5px;height:20px;width:100%}
  3174. .pl-progress-inner-text:after{display:inline-block;content:"";height:100%;vertical-align:middle}
  3175. .pl-ext{display:inline-block;width:44px;background:#999;color:#fff;height:16px;line-height:16px;font-size:12px;border-radius:3px}
  3176. .pl-retry{padding:3px 10px;background:#cc3235;color:#fff;border-radius:3px;cursor:pointer}
  3177. .pl-extra{display:flex;justify-content:center;background-color:${temp.color}15;border-radius:10px;gap:10px}
  3178. .pl-extra:has(>*){margin-top:1.25em;padding:8px 6px}
  3179. .pl-extra>.api.listener-download-all,.pl-extra>.curl.listener-copy,.pl-extra>.aria2.listener-send-rpc,.pl-extra>.bitcomet.listener-copy,.pl-extra>.abdm{flex:1}
  3180. .pl-extra:not(:has(>.api.listener-download-all,>.curl.listener-copy,>.aria2.listener-send-rpc,>.bitcomet.listener-copy,>.abdm))>*{flex:1}
  3181. .pl-btn-primary{color:#ffffff!important;background:${temp.color};border:0;border-radius:50px;cursor:pointer;font-size:12px;outline:none;display:flex;align-items:center;justify-content:center;gap:5px;padding:0.625em 1.1em;transition:opacity.2s,box-shadow.2s;will-change:opacity,box-shadow}
  3182. .pl-btn-primary:hover{opacity:0.8}
  3183. .pl-btn-primary:focus{box-shadow:0 0 0 3px ${temp.color}80}
  3184. .pl-btn-success{background:#55af28}
  3185. .pl-btn-success:focus{box-shadow:0 0 0 3px #55af2880}
  3186. .pl-btn-info{background:#606266}
  3187. .pl-btn-info:focus{box-shadow:0 0 0 3px #60626680}
  3188. .pl-btn-warning{background:#da9328}
  3189. .pl-btn-warning:focus{box-shadow:0 0 0 3px #da932880}
  3190. .pl-btn-danger{background:#cc3235}
  3191. .pl-btn-danger:focus{box-shadow:0 0 0 3px #cc323580}
  3192. .pl-btn-opacity{animation:easeOpacity 1.2s 2;animation-fill-mode:forwards;will-change:opacity}
  3193. @keyframes easeOpacity{ from{opacity:1} 50%{opacity:0.35} to{opacity:1} }
  3194. .pl-button-mini{padding:5px 10px}
  3195. .pl-button,.pl-dropdown-menu{transition:all.2s}
  3196. .pl-button{position:relative}
  3197. .pl-button .pl-dropdown-menu{opacity:0;pointer-events:none;will-change:opacity}
  3198. .pl-button:hover .pl-dropdown-menu{opacity:1;pointer-events:auto}
  3199. .pl-button-init{opacity:0.5;animation:easeInitOpacity 1.2s 5;animation-fill-mode:forwards}
  3200. @keyframes easeInitOpacity{ from{opacity:0.5} 50%{opacity:1} to{opacity:0.5} }
  3201. .pl-dropdown-menu{position:absolute;padding:5px 0;color:${temp.color};background:#fff;z-index:999;min-width:110px;border-radius:5px;box-shadow:0 1px 6px ${temp.color}33;-webkit-box-shadow:0 1px 6px ${temp.color}33;text-align:center;border:none}
  3202. @media (prefers-color-scheme:dark){ .pl-dropdown-menu{color:#fff;background:#222226} }
  3203. .pl-button-mode{color:${temp.color}!important;height:30px;padding:0 10px!important;display:flex;align-items:center;justify-content:center;gap:5px;cursor:pointer;white-space:nowrap;transition:background-color.2s;will-change:background-color}
  3204. @media (prefers-color-scheme:dark){ .pl-dropdown-menu .pl-button-mode{color:#fff!important} }
  3205. .pl-button-mode:hover{background-color:${temp.color}33!important}
  3206. @media (prefers-color-scheme:dark){ .pl-button-mode:hover{color:#fff!important;background:${temp.color}!important} }
  3207. header[style="display:none;"]~.pl-button{display:inline-block;position:fixed;top:0.6em;left:65%;z-index:99999}
  3208. .color-button{background:${temp.color}!important;border-color:${temp.color}!important;border:1px solid ${temp.color}!important;display:inline-flex;transition:background.2s,border-color.2s;will-change:background,border-color}
  3209. .color-button:hover{background:${temp.color}b0!important;border-color:${temp.color}!important}
  3210. .ali-button{background:${temp.color};border:0 solid transparent;font-size:14px;margin-left:20px;padding:8px 16px;position:relative;height:32px;border-radius:100px;display:flex;align-items:center;justify-content:center;color:var(--basic_white);cursor:pointer;transition:background.2s;will-change:background}
  3211. .ali-button:hover{background:${temp.color}D0}
  3212. .ali-btn-icon{vertical-align:-0.2em}
  3213. .mcloud-button{float:left;position:relative;margin:20px 24px 20px 0;width:110px;height:36px;background:${temp.color};border-radius:2px;font-size:14px;color:#fff;line-height:39px;text-align:center;cursor:pointer;will-change:background}
  3214. .mcloud-button:hover{background:${temp.color}b0}
  3215. .mcloud-share-button{display:inline-block;position:relative;font-size:14px;line-height:36px;height:36px;text-align:center;color:#fff;border:1px solid ${temp.color};border-radius:2px;padding:0 24px;background:${temp.color};will-change:background}
  3216. .mcloud-share-button:hover{background:${temp.color}b0}
  3217. .mcloud-btn{background:url("");height:20px;line-height:20px;display:inline-block;background-repeat:no-repeat;background-size:20px 20px;text-indent:25px}
  3218. .tcloud-button{color:#fff;border:1px solid ${temp.color};background:${temp.color};position:relative;height:30px;padding:0 12px;margin-right:12px;font-size:12px;line-height:28px;cursor:pointer;will-change:background,border-color}
  3219. .tcloud-button:hover{border-color:${temp.color}b0;background:${temp.color}b0}
  3220. .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.2s,color.2s,border.2s,box-shadow.2s;color:#fff;background:${temp.color};margin-left:12px;padding:0px 12px;position:relative;cursor:pointer;height:36px;will-change:background}
  3221. .xunlei-button:hover{background:${temp.color}b0}
  3222. .quark-button,.uc-button{padding:0 14px;background:${temp.color}!important;background-color:${temp.color}!important;will-change:background,background-color}
  3223. .uc-btn-icon{width:20px;height:20px;vertical-align:-0.3em}
  3224. .uc-button{padding:10px 20px!important}
  3225. .pl-setting-item{display:flex;align-items:center;justify-content:space-between;margin-top:1em}
  3226. .pl-setting-item > *:nth-child(2){max-width:80%;display:flex;justify-content:space-between;align-items:center}
  3227. .pl-setting-item .pl-setting-item{margin:0;gap:5px}
  3228. .pl-input{padding:8px 10px!important;border:1px solid #c2c2c2;border-radius:5px;font-size:14px!important;margin:0;appearance:auto!important}
  3229. .pl-setting-item > .pl-input{width:80%}
  3230. .init-input{width:90%;margin:0;margin:10px 0;text-align:center;font-family:system-ui,-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Ubuntu,"Helvetica Neue",sans-serif;font-weight:300}
  3231. .pl-tooltip{position:absolute;z-index:110000;display:none;align-items:center;color:#ffffff;background:#333;font-size:12px;line-height:1.3;max-width:600px;border-radius:5px;word-break:break-all;will-change:display,top,left}
  3232. .pl-tooltip>*{padding:5px 10px}
  3233. .pl-tooltip>*:first-child{border:1px solid;border-top-left-radius:5px;border-bottom-left-radius:5px;border-top-color:#333;border-bottom-color:#333;border-left-color:#333;border-right-color:transparent}
  3234. .pl-tooltip>*:last-child{border:1px solid;border-top-right-radius:5px;border-bottom-right-radius:5px;border-top-color:#333;border-bottom-color:#333;border-left-color:transparent;border-right-color:#333}
  3235. .pl-loading-box>div>div{position:absolute;border-radius:50%}
  3236. .pl-loading-box>div>div:nth-child(1){top:9px;left:9px;width:82px;height:82px;background:#ffffff}
  3237. @keyframes load{ 0%{transform:rotate(0deg)} 100%{transform:rotate(360deg)} }
  3238. .pl-loading-box>div>div:nth-child(2){top:14px;left:38px;width:25px;height:25px;background:${temp.color};animation:load 1s linear infinite;transform-origin:12px 36px}
  3239. .pl-loading{width:16px;height:16px;display:inline-block;overflow:hidden;background:none}
  3240. .pl-loading-box{width:100%;height:100%;position:relative;transform:translateZ(0) scale(0.16);backface-visibility:hidden;transform-origin:0 0}
  3241. .pl-loading-box div{box-sizing:content-box}
  3242. .pl-button-save{background-color:${temp.color}!important;color:#fff!important}
  3243. .pl-button-save:hover{background-color:${temp.color}D0!important;color:#fff!important}
  3244. .swal2-container{z-index:100000}
  3245. body.swal2-height-auto{height:inherit}
  3246. [class^="swal2-"],[class*="pl-btn"]{transition:all.2s}
  3247. /* 适配(改)百度网盘会员青春版 */
  3248. a.downloadSubtitle, button.downloadSubtitle{transition:all.2s;background-color:${temp.color}}
  3249. a.downloadSubtitle:hover, button.downloadSubtitle:hover{background-color:${temp.color}D0}
  3250. a.downloadSubtitle:disabled, button.downloadSubtitle:disabled{background-color:${temp.color}D0}
  3251. /* 哪里都没用到的 RGB! */
  3252. @keyframes RGB{ 0%{filter:hue-rotate()} to{filter:hue-rotate(-360deg)} }
  3253. /* Webkit, Opera, IE9, Chrome*/
  3254. ::selection, ::-webkit-selection, ::-moz-selection, ::-ms-selection{background-color:${temp.color}!important;background:${temp.color}!important;color:white!important}
  3255. `);
  3256. },
  3257. /**
  3258. * 初始化引导弹窗
  3259. * @author 油小猴
  3260. * @author hmjz100
  3261. * @description 显示初始化对话框,引导用户进行配置或跳过流程。
  3262. * 支持输入特定数字触发彩蛋,并自动注入默认设置点亮功能。
  3263. * @returns {Promise<void>} 弹窗关闭后返回空值,可能触发页面刷新
  3264. */
  3265. async showInitDialog() {
  3266. let dialog = await Swal.fire({
  3267. ...temp.swalDefault,
  3268. title: `(◍•ᴗ•◍)/ 你好呀`,
  3269. html: `<div class="pl-init-content">
  3270. <p>“这里好像没什么好输入的…”</p>
  3271. <input type="text" autocomplete="off" class="swal2-input init-input" id="init" placeholder="输入内容...">
  3272. <p>
  3273. 但其实…你可以按下下方的 <span style="color:red">红色按钮</span> 跳过流程<br/>
  3274. 或者继续输入一些神秘的 <span class="listener-tip" data-title="乙烯一克,一克一克一克……锕!<br/>      ▃▃▆█▇▄▖<br/>    ▟◤▖    ◥█▎<br/>  ◢◤  ▐     ▐▉<br/> ▗◤  ▂  ▗▖   ▕█▎<br/> ◤ ▗▅▖◥▄ ▀◣    █▊<br/>▐ ▕▎◥▖◣◤     ◢██<br/>█◣ ◥▅█▀     ▐██◤<br/>▐█▙▂      ◢██◤<br/> ◥██◣    ◢▄◤<br/>   ▀██▅▇▀" style="font-style:italic;color:#412300;background-color:#d0b164">“恶臭数字”</span>,解锁隐藏彩蛋~
  3275. </p>
  3276. <p>
  3277. 如果您喜欢这个脚本的话<br/>
  3278. 请支持原作者 <a class="listener-tip pl-a" target="_blank" href="https://www.youxiaohou.com" data-title="${config.base.service.account ? `的微信公众号……<br/><img style='width:250px' src='${config.base.service.account}'>` : ""}"><svg class="pl-icon"><use xlink:href="#pl-icon-si-tampermonkey"></use></svg> 油小猴</a><br/>
  3279. 或给此改版点个 <a class="listener-tip pl-a" target="_blank" href="https://github.com/hmjz100/LinkSwift/" data-title="来看看此项目的 Github 页面吧~"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-star"></use></svg> Star</a>
  3280. </p>
  3281. <p>
  3282. 脚本不仅能精简网盘界面<br/>点亮后还能修改多个网盘的主题色哦!
  3283. </p>
  3284. </div>
  3285. <style>
  3286. .pl-init-content p{margin:10px 0!important}
  3287. </style>
  3288. `,
  3289. showCloseButton: true,
  3290. showDenyButton: true,
  3291. denyButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-unlock-keyhole"></use></svg> 懒得输入...我要直接点亮!',
  3292. allowOutsideClick: false,
  3293. });
  3294. if (dialog.isDenied) {
  3295. message.warning("正在“注入”设置项目...");
  3296. await base.sleep(2500);
  3297. let list = base.getValue("setting_init");
  3298. list.code = config.base.num;
  3299. list.license = config.base.license;
  3300. base.setValue("setting_init", list);
  3301. message.success("“注入”成功啦!");
  3302. await base.sleep(1500);
  3303. location.reload();
  3304. };
  3305. if (dialog.isConfirmed) {
  3306. if ($('#init').val() === '114514' || $('#init').val() === '1919810' || $('#init').val() === '1145141919810') {
  3307. await Swal.fire({
  3308. ...temp.swalDefault,
  3309. icon: 'error',
  3310. title: '1145141919810',
  3311. html: '<span>homo特有的数字当然不行啦<br/>哼哼哼啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊</span>',
  3312. timer: 4000,
  3313. imageUrl: 'https://pic1.zhimg.com/v2-1b97a088e156c015108dec663bba8b04.avis',
  3314. allowOutsideClick: false,
  3315. timerProgressBar: true,
  3316. showConfirmButton: false,
  3317. showDenyButton: true,
  3318. denyButtonText: '哼哼哼啊啊啊啊啊啊啊啊啊啊',
  3319. });
  3320. message.info("成就:你触发了一个homo特有的彩蛋!");
  3321. await base.sleep(4000)
  3322. Swal.fire({
  3323. ...temp.swalDefault,
  3324. title: '1145141919810',
  3325. text: 'homo特有的数字当然不行啦...吗?',
  3326. icon: 'question',
  3327. imageUrl: 'https://lh1.hetaousercontent.com/img/7d4c1c0b4adb0e95.jpg',
  3328. showConfirmButton: false,
  3329. allowOutsideClick: false,
  3330. });
  3331. await base.sleep(3000)
  3332. let list = base.getValue("setting_init");
  3333. list.code = config.base.num;
  3334. list.license = config.base.license;
  3335. base.setValue("setting_init", list);
  3336. message.success("成就:哼哼哼啊啊啊啊啊啊啊啊地注入成功(喜)");
  3337. await base.sleep(1500)
  3338. location.reload();
  3339. } else {
  3340. await this.showInitDialog();
  3341. return;
  3342. };
  3343. }
  3344. },
  3345. /**
  3346. * 显示主对话框
  3347. * @author 油小猴
  3348. * @author hmjz100
  3349. * @description 使用 SweetAlert2 显示一个自定义样式的对话框,用于展示信息或操作提示。
  3350. * @param {string} title - 对话框标题
  3351. * @param {string} html - 对话框内容的 HTML 字符串
  3352. * @param {string} footer - 对话框底部说明文字
  3353. */
  3354. showMainDialog(title, html, footer) {
  3355. Swal.fire({
  3356. ...temp.swalDefault,
  3357. title,
  3358. html,
  3359. footer: `${footer}<p>${config.base.dom.footer}</p>`,
  3360. customClass: {
  3361. popup: 'pl-popup',
  3362. header: 'pl-header',
  3363. title: 'pl-title',
  3364. closeButton: 'pl-close',
  3365. content: 'pl-content',
  3366. input: 'pl-input',
  3367. footer: 'pl-footer'
  3368. },
  3369. confirmButtonText: `<svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg> 关闭`,
  3370. showCloseButton: true,
  3371. allowOutsideClick: false,
  3372. allowEscapeKey: false,
  3373. allowEnterKey: false,
  3374. willClose: () => {
  3375. base._resetAllData();
  3376. },
  3377. }).then(() => {
  3378. base._resetAllData();
  3379. });
  3380. },
  3381. /**
  3382. * 等待指定元素加载完成并执行回调
  3383. * @author hmjz100
  3384. * @description 监听 DOM 元素是否出现,若未出现则每隔一段时间重试,直到找到为止。
  3385. * 支持在 iframe 内部查找元素,适用于异步加载场景。
  3386. * @param {string} selectorElem - 要等待的目标元素选择器
  3387. * @param {Function} actionFunction - 找到元素后执行的回调函数,接收 jQuery 元素作为参数,返回 true 可以不再继续寻找
  3388. * @param {boolean} [bWaitOnce=false] - 是否只执行一次回调,默认为 false,如果不设置为 true 的话需要自行判断是否对元素进行操作
  3389. * @param {string} [iframeSelector] - 若目标元素位于 iframe 中,传入 iframe 的选择器
  3390. * @param {string} [controlKey] - 控制唯一性的键名,用于避免重复处理
  3391. */
  3392. waitForKeyElements(selectorElem, actionFunction, bWaitOnce, iframeSelector, controlKey) {
  3393. // 初始化管理器
  3394. const manager = this.waitForKeyElements.manager || (
  3395. this.waitForKeyElements.manager = {
  3396. observers: new WeakMap(),
  3397. tasks: new Map(),
  3398. instanceCounter: 0
  3399. }
  3400. );
  3401. const targetDoc = iframeSelector
  3402. ? $(iframeSelector).get(0)?.contentDocument
  3403. : document;
  3404. if (!targetDoc) return; // 无效文档直接返回
  3405. // 生成唯一控制键
  3406. controlKey = controlKey || `wkfe_${manager.instanceCounter++}`;
  3407. // 清理重复任务
  3408. const existingTask = manager.tasks.get(controlKey);
  3409. if (existingTask) {
  3410. existingTask.observer.disconnect();
  3411. manager.tasks.delete(controlKey);
  3412. }
  3413. // 创建MutationObserver回调
  3414. const processElements = () => {
  3415. const elements = $(selectorElem, targetDoc);
  3416. let foundActive = false;
  3417. elements.each((i, el) => {
  3418. const jEl = $(el);
  3419. const isProcessed = jEl.data(controlKey);
  3420. if (isProcessed) return true; // 跳过已处理元素
  3421. const cancelAction = actionFunction(jEl);
  3422. if (cancelAction) {
  3423. foundActive = true;
  3424. } else if (bWaitOnce) {
  3425. jEl.data(controlKey, true); // 标记已处理
  3426. }
  3427. });
  3428. // 一次性任务且找到有效元素时清理
  3429. if (bWaitOnce && foundActive) {
  3430. observer.disconnect();
  3431. manager.tasks.delete(controlKey);
  3432. }
  3433. };
  3434. // 创建Observer实例
  3435. const observer = new MutationObserver(processElements);
  3436. // 配置并启动观察
  3437. observer.observe(targetDoc.documentElement, {
  3438. childList: true,
  3439. subtree: true,
  3440. attributes: true,
  3441. characterData: true
  3442. });
  3443. // 注册任务
  3444. manager.tasks.set(controlKey, {
  3445. observer,
  3446. targetDoc
  3447. });
  3448. // 立即执行初始检查
  3449. processElements();
  3450. },
  3451. /**
  3452. * 状态工厂
  3453. * @author 油小猴
  3454. * @author hmjz100
  3455. * @description 接受被监听的 DOM 元素的状态,根据状态确定元素是谁
  3456. * @param {Event} event - 元素状态
  3457. */
  3458. _EventFactory(event) {
  3459. let target = $(event.target);
  3460. let item = target.parents('.pl-item');
  3461. let link = item.find('.pl-item-link.enhance');
  3462. let directLink = item.find('.pl-item-link.normal');
  3463. let progress = item.find('.pl-item-progress');
  3464. let tip = item.find('.pl-item-tip');
  3465. let copy = item.find('.pl-item-copy');
  3466. let back = item.find('.pl-progress-back');
  3467. let stop = item.find('.pl-progress-stop');
  3468. return {
  3469. item, link, directLink, progress, tip, copy, back, stop, target,
  3470. };
  3471. }
  3472. };
  3473. /**
  3474. * 百度网盘
  3475. * @author 油小猴
  3476. * @author hmjz100
  3477. */
  3478. let $baidu = {
  3479. async getToken() {
  3480. try {
  3481. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  3482. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取授权状态~</div>`);
  3483. // 获取授权状态
  3484. let authorize = await base.getFinalUrl(config.$baidu.api.getAccessToken, undefined, true);
  3485. let accessToken = "";
  3486. // 判断授权情况
  3487. if (authorize.includes('authorize')) {
  3488. $doc.find('.loading-popup .loading-title').html(`授权获取中`);
  3489. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取授权页面~</div>`);
  3490. // 没授权,先获取授权的页面
  3491. let html = await base.get(config.$baidu.api.getAccessToken, undefined, 'text');
  3492. // 提取页面的发送确认授权的参数
  3493. let bdstoken = html.match(/name="bdstoken"\s+value="([^"]+)"/)?.[1];
  3494. let client_id = html.match(/name="client_id"\s+value="([^"]+)"/)?.[1];
  3495. let data = {
  3496. grant_permissions_arr: 'netdisk',
  3497. bdstoken: bdstoken,
  3498. client_id: client_id,
  3499. response_type: "token",
  3500. display: "page",
  3501. grant_permissions: "basic,netdisk"
  3502. };
  3503. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在自动确认授权~</div>`);
  3504. // 发送请求达到自动进行授权
  3505. await base.post(config.$baidu.api.getAccessToken, base.stringify(data), { "Content-Type": "application/x-www-form-urlencoded" });
  3506. // 再次获取授权状态
  3507. let res2 = await base.getFinalUrl(config.$baidu.api.getAccessToken, undefined, true);
  3508. accessToken = res2.match(/access_token=([^&]+)/)?.[1];
  3509. } else if (authorize.includes('access_token=')) {
  3510. accessToken = authorize.match(/access_token=([^&]+)/)?.[1];
  3511. }
  3512. // 统一处理令牌结果
  3513. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  3514. if (!!accessToken) {
  3515. $doc.find('.loading-popup .swal2-html-container').html(`<div>授权成功,令牌已缓存~</div>`);
  3516. base.setValue('baidu_access_token', accessToken);
  3517. return accessToken;
  3518. } else return "";
  3519. } catch (error) {
  3520. return "";
  3521. }
  3522. },
  3523. addPageListener() {
  3524. $doc.on('mouseenter mouseleave click', '.pl-button.g-dropdown-button', function (e) {
  3525. if (e.type === 'mouseleave') {
  3526. $(e.currentTarget).removeClass('button-open');
  3527. } else {
  3528. $(e.currentTarget).addClass('button-open');
  3529. $(e.currentTarget).find('.pl-dropdown-menu').show();
  3530. }
  3531. });
  3532. $doc.on('mouseleave', '.pl-button.g-dropdown-button .pl-dropdown-menu', function (e) {
  3533. $(e.currentTarget).hide();
  3534. });
  3535. $doc.on('click', '.pl-button-mode', async function (e) {
  3536. temp.mode = e.currentTarget.dataset.mode;
  3537. if (!temp.mode) return;
  3538. $baidu.getLink();
  3539. });
  3540. $doc.on('click', '.pl-button-save', async function (e) {
  3541. e.preventDefault();
  3542. let selectList = $baidu.getSelectedList();
  3543. if (selectList.length === 0) {
  3544. return message.error('提示:<br/>请勾选要保存到网盘的文件哦~');
  3545. }
  3546. message.info('提示:<br/>因网盘限制,请保存到自己网盘后再去下载哦~');
  3547. await base.sleep(500);
  3548. document.querySelector('.tools-share-save-hb').click();
  3549. });
  3550. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  3551. e.preventDefault();
  3552. let o = base._EventFactory(e);
  3553. let $width = o.item.find('.pl-progress-inner');
  3554. let $text = o.item.find('.pl-progress-inner-text');
  3555. let filename = o.link[0].dataset.filename;
  3556. let index = o.link[0].dataset.index;
  3557. let size = Number(o.link[0].dataset.size) || 0;
  3558. let originalHtml = o.link.html();
  3559. base._resetData(index);
  3560. base.get(o.link[0].dataset.link, { "User-Agent": config.$baidu.api.ua.downloadLink, "Origin": "", "Referer": "", "DNT": "" }, 'blob', { filename, index });
  3561. let startTime = Date.now();
  3562. let prevLoaded = 0;
  3563. let prevTime = startTime;
  3564. temp.ins[index] = setInterval(async () => {
  3565. let prog = +temp.progress[index] || 0;
  3566. let isIDM = !!temp.idm[index];
  3567. if (isIDM) {
  3568. // IDM 捕获处理逻辑
  3569. o.tip.hide();
  3570. o.progress.hide();
  3571. o.copy.show();
  3572. o.directLink.show();
  3573. o.link
  3574. .text('链接已被IDM捕获~请查看IDM下载窗口哦!')
  3575. .animate({ opacity: '0.5' }, "slow")
  3576. .show();
  3577. clearInterval(temp.ins[index]);
  3578. await base.sleep(2000);
  3579. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  3580. temp.idm[index] = false;
  3581. return;
  3582. }
  3583. let currentTime = Date.now();
  3584. let elapsedTime = currentTime - startTime;
  3585. let loaded = prog * size / 100;
  3586. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  3587. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  3588. // 计算剩余时间(保护除零)
  3589. let totalProgress = Math.max(prog / 100, 0.01);
  3590. let totalElapsedSeconds = elapsedTime / 1000;
  3591. let estTotalTime = totalElapsedSeconds / totalProgress;
  3592. let remainingTime = estTotalTime - totalElapsedSeconds;
  3593. // 更新界面状态
  3594. o.link.hide();
  3595. o.directLink.hide();
  3596. o.tip.hide();
  3597. o.stop.show();
  3598. o.copy.hide();
  3599. o.progress.show();
  3600. // 更新进度条
  3601. $width.css('width', `${prog}%`);
  3602. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  3603. // 更新历史值
  3604. prevLoaded = loaded;
  3605. prevTime = currentTime;
  3606. // 下载完成
  3607. if (prog >= 100) {
  3608. await base.sleep(1000);
  3609. clearInterval(temp.ins[index]);
  3610. temp.progress[index] = 0;
  3611. o.item.find('.pl-progress-stop').hide();
  3612. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  3613. o.back.show();
  3614. await base.sleep(3000);
  3615. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  3616. }
  3617. }, 500);
  3618. });
  3619. $doc.on('click', '.listener-aria2-download', async function (e) {
  3620. let target = $(e.currentTarget);
  3621. if (target.attr('data-processing') === 'true') return;
  3622. target.attr('data-processing', 'true');
  3623. let originalHtml = target.html();
  3624. target.find(".pl-icon").remove();
  3625. target.find('.pl-loading').remove();
  3626. target.prepend(base.createLoading());
  3627. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"), [`User-Agent:${config.$baidu.api.ua.downloadLink}`]);
  3628. if (res === 'success') {
  3629. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  3630. } else {
  3631. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  3632. }
  3633. await base.sleep(3000);
  3634. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  3635. });
  3636. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  3637. let target = $(e.currentTarget);
  3638. if (target.attr('data-processing') === 'true') return;
  3639. target.attr('data-processing', 'true');
  3640. let originalHtml = target.html();
  3641. target.find(".pl-icon").remove();
  3642. target.find('.pl-loading').remove();
  3643. target.prepend(base.createLoading());
  3644. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"), { "user_agent": config.$baidu.api.ua.downloadLink });
  3645. if (res === 'success') {
  3646. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  3647. } else {
  3648. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  3649. }
  3650. await base.sleep(3000);
  3651. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  3652. });
  3653. $doc.on('click', '.listener-abdm-download', async function (e) {
  3654. let target = $(e.currentTarget);
  3655. if (target.attr('data-processing') === 'true') return;
  3656. target.attr('data-processing', 'true');
  3657. let originalHtml = target.html();
  3658. target.find(".pl-icon").remove();
  3659. target.find('.pl-loading').remove();
  3660. target.prepend(base.createLoading());
  3661. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), { "User-Agent": config.$baidu.api.ua.downloadLink });
  3662. if (res === 'success') {
  3663. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  3664. } else {
  3665. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  3666. }
  3667. await base.sleep(3000);
  3668. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  3669. });
  3670. },
  3671. greenerPage() {
  3672. temp.pege = $baidu.detectPage();
  3673. base.waitForKeyElements(".wp-s-header-user__vip-center", function (tag) {
  3674. tag.remove();
  3675. }, true);
  3676. base.waitForKeyElements(".wp-s-header-user__create-team-content", function (tag) {
  3677. tag.remove();
  3678. }, true);
  3679. base.waitForKeyElements(".app-user-vip-center-box.vip-center-type-2", function (tag) {
  3680. tag.remove();
  3681. }, true);
  3682. base.waitForKeyElements(".wp-s-header__vip-btn-tip", function (tag) {
  3683. tag.fadeOut();
  3684. }, true);
  3685. base.waitForKeyElements(".app-user-vip-center-tip", function (tag) {
  3686. tag.fadeOut();
  3687. }, true);
  3688. base.waitForKeyElements("#web-header-text-s-45", function (tag) {
  3689. tag.fadeOut();
  3690. }, true);
  3691. base.waitForKeyElements(".wp-s-header__vip-btn", function (tag) {
  3692. tag.text("会员中心")
  3693. }, true);
  3694. base.waitForKeyElements(".KQcHyA", function (tag) {
  3695. tag.text("会员中心")
  3696. }, true);
  3697. base.waitForKeyElements(".gOIbzPb", function (tag) {
  3698. tag.fadeOut();
  3699. }, true);
  3700. base.waitForKeyElements(".wp-s-header-user__create-team-title", function (tag) {
  3701. tag.fadeOut();
  3702. }, true);
  3703. base.waitForKeyElements(".web-header-ad-item", function (tag) {
  3704. tag.fadeOut();
  3705. });
  3706. base.waitForKeyElements(".wp-s-header__game-entry", function (tag) {
  3707. tag.fadeOut();
  3708. }, true)
  3709. base.waitForKeyElements(".bd-aside-ad", function (tag) {
  3710. tag.fadeOut();
  3711. }, true)
  3712. base.waitForKeyElements(".btn-img-tips", function (tag) {
  3713. tag.fadeOut();
  3714. }, true)
  3715. base.waitForKeyElements(".nd-operate-guidance", function (tag) {
  3716. tag.fadeOut();
  3717. }, true)
  3718. base.waitForKeyElements(".module-operation-content", function (tag) {
  3719. tag.fadeOut();
  3720. document.querySelector(".operate-guide-close").click();
  3721. document.querySelector(".module-canvas").click();
  3722. }, true)
  3723. base.waitForKeyElements("[class*='module-'][class*='-box']:not(.module-box), [class*='module-'][class*='-mask']", function (tag) {
  3724. tag.fadeOut();
  3725. tag.find(".close-mask").click();
  3726. }, true)
  3727. base.waitForKeyElements(".newIcon", function (tag) {
  3728. tag.fadeOut();
  3729. }, true);
  3730. base.waitForKeyElements(".u-badge__content.is-dot", function (tag) {
  3731. tag.fadeOut();
  3732. }, true);
  3733. base.waitForKeyElements(".wp-side-options.g-clearfix", function (tag) {
  3734. tag.fadeOut();
  3735. }, true);
  3736. base.waitForKeyElements(".wp-s-header-user__drop-channel", function (tag) {
  3737. tag.fadeOut();
  3738. }, true);
  3739. base.waitForKeyElements(".app-download", function (tag) {
  3740. tag.fadeOut();
  3741. }, true);
  3742. base.waitForKeyElements('.g-button[title*="手机"]', function (tag) {
  3743. tag.fadeOut();
  3744. }, true)
  3745. base.waitForKeyElements('.yike-entrance', function (tag) {
  3746. tag.remove();
  3747. }, true)
  3748. base.waitForKeyElements("a.tools__item", function (tag) {
  3749. if (tag.attr("linked")) return;
  3750. if (tag.attr('href')) {
  3751. try {
  3752. let url = new URL(tag.closest('a').attr('href'));
  3753. url.search = "";
  3754. url.hash = url.hash.replace(/\?(.*?)(#|$)/, '$2')
  3755. tag.attr('href', url.href)
  3756. } catch (e) { }
  3757. }
  3758. tag.attr("linked", true)
  3759. }, true);
  3760. base.waitForKeyElements("p.wp-s-aside-nav__main-item-text", function (tag) {
  3761. if (tag.attr("linked")) return;
  3762. if (tag.closest('a').attr('href')) {
  3763. try {
  3764. let url = new URL(tag.closest('a').attr('href'));
  3765. url.search = "";
  3766. url.hash = url.hash.replace(/\?(.*?)(#|$)/, '$2')
  3767. tag.closest('a').attr('href', url.href)
  3768. } catch (e) { }
  3769. }
  3770. if (tag.is(":contains('插件'), :contains('相册'), :contains('笔记')") && tag.closest('a').attr('target') !== "_blank") {
  3771. tag.closest('a').fadeOut();
  3772. } else {
  3773. tag.text(tag.text().replace("百度", ""));
  3774. }
  3775. tag.attr("linked", true)
  3776. }, true);
  3777. base.waitForKeyElements('dd[node-type="header-link"]', function (tag) {
  3778. tag.children().each((index, element) => {
  3779. let tag = $(element);
  3780. if (!tag.attr("node-type")) return;
  3781. let type = tag.attr("node-type");
  3782. if (
  3783. type !== "disk-home" &&
  3784. type !== "mbox-homepage" &&
  3785. type !== "find-apps"
  3786. ) {
  3787. tag.fadeOut();
  3788. }
  3789. });
  3790. }, true);
  3791. base.waitForKeyElements(".__yunguanjia", function (tag) {
  3792. tag.html(`<div class="yunguanjia-list __yunguanjia row g-clearfix _item sel">
  3793. <span type="radio" class="radio-box _radioInput __yunguanjiaRadio">
  3794. <span class="device-name">添加我的电脑</span>
  3795. </span>
  3796. <div class="__yunguanjiaTips radio-tips" style="display:block;">
  3797. 用电脑下载并登录最新百度网盘客户端,即自动完成添加。
  3798. <a href="//pan.baidu.com/download" target="_blank">下载百度网盘客户端</a>
  3799. <br/>由 <a href="https://github.com/hmjz100/LinkSwift/" target="_blank">LinkSwift</a> 修复该选项
  3800. </div>
  3801. </div>`);
  3802. }, true)
  3803. // 美化分享页面
  3804. if (temp.pege === 'share') {
  3805. base.waitForKeyElements(`iframe[src^="/buy/ad"]`, function (tag) {
  3806. tag.fadeOut();
  3807. }, true)
  3808. base.addStyle(`${mount}-baiduShare`, 'style', `
  3809. body, .theme-white.init-new, #layoutApp{
  3810. background-color:#DCEFFE!important;
  3811. background:#DCEFFE url(https://nd-static.bdstatic.com/m-static/disk-share/widget/pageModule/init-new/image/init-bg_1708266.png) no-repeat center center;
  3812. }
  3813. #bd-main .bd-left{
  3814. background:#ffffffC0;
  3815. border-radius:10px;
  3816. }
  3817. iframe[src="/buy/ad/home"]{
  3818. display:none!important;
  3819. }
  3820. `, `.${mount}`);
  3821. base.waitForKeyElements(`.KPDwCE`, function (tag) {
  3822. tag.css('background', 'transparent');
  3823. }, true);
  3824. base.waitForKeyElements('.share-list .KPDwCE .AuPKyz', function (tag) {
  3825. tag.css('background', 'transparent');
  3826. }, true);
  3827. base.waitForKeyElements(`#layoutMain`, function (tag) {
  3828. tag.css({ "border-radius": "24px" });
  3829. }, true)
  3830. base.waitForKeyElements(".frame-content", function (tag) {
  3831. tag.css({ "margin": "auto" });
  3832. }, true)
  3833. }
  3834. },
  3835. beautifyPage() {
  3836. if ($baidu.detectPage() !== 'home') {
  3837. base.adaptiveThemeOverride([
  3838. ['#717fff', temp.color],
  3839. ['#717FFF', temp.color],
  3840. ['#06a8ff', temp.color],
  3841. ['#06A8FF', temp.color],
  3842. ['#06a7ff', temp.color],
  3843. ['#06A7FF', temp.color],
  3844. ['#dcdfe6', temp.color],
  3845. ['#DCDFE6', temp.color],
  3846. ['#0095ff', temp.color],
  3847. ['#0095FF', temp.color],
  3848. ['#09aaff', temp.color],
  3849. ['#09AAFF', temp.color],
  3850. ['#0ca6ff', temp.color],
  3851. ['#0CA6FF', temp.color],
  3852. ['#5040ff', temp.color],
  3853. ['#5040FF', temp.color],
  3854. ['#454d5a', temp.color],
  3855. ['#454D5A', temp.color],
  3856. ['#a2abbd', temp.color],
  3857. ['#A2ABBD', temp.color],
  3858. ['#030b1a', temp.color],
  3859. ['#030B1A', temp.color],
  3860. ['#afb3bf', temp.color],
  3861. ['#AFB3BF', temp.color],
  3862. ['#ff436a', temp.color],
  3863. ['#FF436A', temp.color],
  3864. ['#03081a', temp.color],
  3865. ['#03081A', temp.color],
  3866. ['#2974b6', temp.color],
  3867. ['#2974B6', temp.color],
  3868. ['#0596e6', temp.color],
  3869. ['#0596E6', temp.color],
  3870. ['#C3EAFF', temp.color],
  3871. ['#c0d9fe', `${temp.color}50`],
  3872. ['#0098EA', `${temp.color}D0`],
  3873. ['#38b9ff', `${temp.color}D0`],
  3874. ['#38B9FF', `${temp.color}D0`],
  3875. ['#42d8ff', `${temp.color}D0`],
  3876. ['#42D8FF', `${temp.color}D0`],
  3877. ['#a48dff', `${temp.color}D0`],
  3878. ['#A48DFF', `${temp.color}D0`],
  3879. ['#6b79f2', `${temp.color}D0`],
  3880. ['#6B79F2', `${temp.color}D0`],
  3881. ['#9c86f2', `${temp.color}90`],
  3882. ['#9C86F2', `${temp.color}90`],
  3883. ['#83d3ff', `${temp.color}90`],
  3884. ['#83D3FF', `${temp.color}90`],
  3885. ['#C4D8F4', `${temp.color}90`],
  3886. ['#fafafc', `${temp.color}20`],
  3887. ['#FAFAFC', `${temp.color}20`],
  3888. ['#f5fbff', `${temp.color}20`],
  3889. ['#F5FBFF', `${temp.color}20`],
  3890. ['#b4e5ff', `${temp.color}20`],
  3891. ['#B4E5FF', `${temp.color}20`],
  3892. ['#f0faff', `${temp.color}20`],
  3893. ['#F0FAFF', `${temp.color}20`],
  3894. ['#c4d8f4', `${temp.color}20`],
  3895. ['#f1f3f8', `${temp.color}15`],
  3896. ['#F1F3F8', `${temp.color}15`],
  3897. ['#f2faff', `${temp.color}10`],
  3898. ['#F2FAFF', `${temp.color}10`],
  3899. ['#eef9fe', `${temp.color}10`],
  3900. ['#EEF9FE', `${temp.color}10`],
  3901. ['#f7f9fc', `${temp.color}10`],
  3902. ['#F7F9FC', `${temp.color}10`],
  3903. ['#f5f6fa', `${temp.color}10`],
  3904. ['#F5F6FA', `${temp.color}10`],
  3905. ['#b4e5ff', `${temp.color}10`],
  3906. ['#B4E5FF', `${temp.color}10`],
  3907. ['#e6f6ff', `${temp.color}10`],
  3908. ['#E6F6FF', `${temp.color}10`],
  3909. ['0,149,255', base.hexToRgba(temp.color)],
  3910. ['30, 175, 255', base.hexToRgba(temp.color)],
  3911. ['6, 167, 255, 0.1', base.hexToRgba(`${temp.color}1a`)],
  3912. ['6,167,255,.1', base.hexToRgba(`${temp.color}1a`)],
  3913. ['6,167,255,.23', base.hexToRgba(`${temp.color}3b`)],
  3914. ['164,141,255,.2', base.hexToRgba(`${temp.color}30`)],
  3915. ['196,182,255,.2', base.hexToRgba(`${temp.color}20`)],
  3916. ['113,127,255,.2', base.hexToRgba(`${temp.color}40`)],
  3917. ['3,8,26,.6', base.hexToRgba(`${temp.color}D0`)],
  3918. ['255,32,102,.4', base.hexToRgba(`${temp.color}66`)],
  3919. ['72,166,248,.7', base.hexToRgba(`${temp.color}66`)],
  3920. ]);
  3921. };
  3922. base.addStyle(`${mount}-baidu`, 'style', `
  3923. #layoutMain,.DxdbeCb{border-radius:10px;border-bottom-left-radius:0;border-bottom-right-radius:0;background:#ffffffA0!important}
  3924. .KPDwCE,
  3925. .DxdbeCb .OFaPaO .tanwePYr,
  3926. .xGLMIab .fufHyA:hover,
  3927. .module-search-timeline .form-box
  3928. {background:#ffffffA0!important}
  3929. .KPDwCE .JDeHdxb,
  3930. .NHcGw .AuPKyz,
  3931. .xGLMIab .tvPMvPb,
  3932. .xGLMIab .FcQMwt,
  3933. .cazEfA .yfHIsP,
  3934. .hscjZ4QL .bbxnZ0Bq .ehnyLxWZ span,
  3935. .module-topToolBar,
  3936. .module-timeline-view .timeline-title-curday
  3937. {background:transparent!important;border-bottom:0}
  3938. .MdLxwM{background :#fff!important}
  3939. .aside-absolute-container{position:absolute!important}
  3940. .aside-absolute-container .QGOvsxb .remainingSpaceUi_span{background:#8af248!important;border-radius:10px 0 0 10px;border-right:#fff 1px solid;border-bottom:#fff 1px solid}
  3941. .xtJbHcb .CDaavKb .KQcHyA{background:rgb(244,207,0)!important;padding:8px 15px}
  3942. .xtJbHcb .web-header-nav-new-version-inner{background:${temp.color}!important;padding:8px 15px;line-height:15px;width:auto;height:auto}
  3943. a{transition:all.2s!important}
  3944. #bd-main .bd-left{margin:auto!important}
  3945. .verify-input input{padding-left:0!important;text-align:center!important}
  3946. .verify-input input:focus{border:2px solid ${temp.color}!important}
  3947. [data-theme=light] .vp-video-page-card .vp-video-page-card__video-detail{color:#030b1a}
  3948. dt.level-1{background:#fd6d65!important}
  3949. dt.level-2{background:#f3a723!important}
  3950. dt.level-1 i.desc-arrow{border-bottom:10px solid #dd6966!important}
  3951. dt.level-2 i.desc-arrow{border-bottom:10px solid #d29633!important}
  3952. `, `.${mount}`);
  3953. base.adaptiveThemeOverride([
  3954. ['#717fff', temp.color],
  3955. ['#717FFF', temp.color],
  3956. ['#06a8ff', temp.color],
  3957. ['#06A8FF', temp.color],
  3958. ['#06a7ff', temp.color],
  3959. ['#06A7FF', temp.color],
  3960. ['#dcdfe6', temp.color],
  3961. ['#DCDFE6', temp.color],
  3962. ['#0095ff', temp.color],
  3963. ['#0095FF', temp.color],
  3964. ['#09aaff', temp.color],
  3965. ['#09AAFF', temp.color],
  3966. ['#0ca6ff', temp.color],
  3967. ['#0CA6FF', temp.color],
  3968. ['#5040ff', temp.color],
  3969. ['#5040FF', temp.color],
  3970. ['#454d5a', temp.color],
  3971. ['#454D5A', temp.color],
  3972. ['#a2abbd', temp.color],
  3973. ['#A2ABBD', temp.color],
  3974. ['#030b1a', temp.color],
  3975. ['#030B1A', temp.color],
  3976. ['#afb3bf', temp.color],
  3977. ['#AFB3BF', temp.color],
  3978. ['#ff436a', temp.color],
  3979. ['#FF436A', temp.color],
  3980. ['#03081a', temp.color],
  3981. ['#03081A', temp.color],
  3982. ['#2974b6', temp.color],
  3983. ['#2974B6', temp.color],
  3984. ['#0596e6', temp.color],
  3985. ['#0596E6', temp.color],
  3986. ['#C3EAFF', temp.color],
  3987. ['#c0d9fe', `${temp.color}50`],
  3988. ['#0098EA', `${temp.color}D0`],
  3989. ['#38b9ff', `${temp.color}D0`],
  3990. ['#38B9FF', `${temp.color}D0`],
  3991. ['#42d8ff', `${temp.color}D0`],
  3992. ['#42D8FF', `${temp.color}D0`],
  3993. ['#a48dff', `${temp.color}D0`],
  3994. ['#A48DFF', `${temp.color}D0`],
  3995. ['#6b79f2', `${temp.color}D0`],
  3996. ['#6B79F2', `${temp.color}D0`],
  3997. ['#9c86f2', `${temp.color}90`],
  3998. ['#9C86F2', `${temp.color}90`],
  3999. ['#83d3ff', `${temp.color}90`],
  4000. ['#83D3FF', `${temp.color}90`],
  4001. ['#C4D8F4', `${temp.color}90`],
  4002. ['#fafafc', `${temp.color}20`],
  4003. ['#FAFAFC', `${temp.color}20`],
  4004. ['#f5fbff', `${temp.color}20`],
  4005. ['#F5FBFF', `${temp.color}20`],
  4006. ['#b4e5ff', `${temp.color}20`],
  4007. ['#B4E5FF', `${temp.color}20`],
  4008. ['#f0faff', `${temp.color}20`],
  4009. ['#F0FAFF', `${temp.color}20`],
  4010. ['#c4d8f4', `${temp.color}20`],
  4011. ['#f1f3f8', `${temp.color}15`],
  4012. ['#F1F3F8', `${temp.color}15`],
  4013. ['#f2faff', `${temp.color}10`],
  4014. ['#F2FAFF', `${temp.color}10`],
  4015. ['#eef9fe', `${temp.color}10`],
  4016. ['#EEF9FE', `${temp.color}10`],
  4017. ['#f7f9fc', `${temp.color}10`],
  4018. ['#F7F9FC', `${temp.color}10`],
  4019. ['#f5f6fa', `${temp.color}10`],
  4020. ['#F5F6FA', `${temp.color}10`],
  4021. ['#b4e5ff', `${temp.color}10`],
  4022. ['#B4E5FF', `${temp.color}10`],
  4023. ['#e6f6ff', `${temp.color}10`],
  4024. ['#E6F6FF', `${temp.color}10`],
  4025. ['0,149,255', base.hexToRgba(temp.color)],
  4026. ['30, 175, 255', base.hexToRgba(temp.color)],
  4027. ['6, 167, 255, 0.1', base.hexToRgba(`${temp.color}1a`)],
  4028. ['6,167,255,.1', base.hexToRgba(`${temp.color}1a`)],
  4029. ['6,167,255,.23', base.hexToRgba(`${temp.color}3b`)],
  4030. ['164,141,255,.2', base.hexToRgba(`${temp.color}30`)],
  4031. ['196,182,255,.2', base.hexToRgba(`${temp.color}20`)],
  4032. ['113,127,255,.2', base.hexToRgba(`${temp.color}40`)],
  4033. ['3,8,26,.6', base.hexToRgba(`${temp.color}D0`)],
  4034. ['255,32,102,.4', base.hexToRgba(`${temp.color}66`)],
  4035. ['72,166,248,.7', base.hexToRgba(`${temp.color}66`)],
  4036. ], "other");
  4037. },
  4038. addButton() {
  4039. base.waitForKeyElements(config.$baidu.mount.home, (element) => {
  4040. temp.pege = $baidu.detectPage();
  4041. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  4042. let $button = $(`<div class="g-dropdown-button pl-button">
  4043. <div class="g-button g-button-blue color-button"><span class="g-button-right"><em class="icon icon-download" style="color:#fff;"></em><span class="text" style="width:60px;">下载助手</span></span></div>
  4044. <div class="menu" style="color:${temp.color};border-color:${temp.color};width:auto;z-index:41;">
  4045. <div class="g-button-menu pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg> API 下载</div>
  4046. <div class="g-button-menu pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg> cURL 下载</div>
  4047. <div class="g-button-menu pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg> Aria2 下载</div>
  4048. <div class="g-button-menu pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg> 彗星下载</div>
  4049. <div class="g-button-menu pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg> ABDM 下载</div>
  4050. <div class="g-button-menu pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg> 助手设置</div>
  4051. <div class="g-button-menu pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg> 助手美化</div>
  4052. <div class="g-button-menu pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg> 更新日志</div>
  4053. </div>
  4054. </div>`);
  4055. element.prepend($button);
  4056. })
  4057. base.waitForKeyElements(config.$baidu.mount.main, (element) => {
  4058. temp.pege = $baidu.detectPage();
  4059. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'main') return;
  4060. let $button = $(`<div class="wp-s-agile-tool-bar__h-group pl-button">
  4061. <div class="wp-s-agile-tool-bar__h-action is-need-left-sep is-main color-button">
  4062. <button type="button" class="u-button nd-file-list-toolbar-action-item u-button--primary u-button--small is-round is-has-icon pl-button color-button">
  4063. <i class="u-icon u-icon-download"></i>
  4064. <span>下载助手</span>
  4065. </button>
  4066. <ul class="dropdown-list nd-common-float-menu pl-dropdown-menu">
  4067. <li class="sub cursor-p pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  4068. <li class="sub cursor-p pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  4069. <li class="sub cursor-p pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  4070. <li class="sub cursor-p pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  4071. <li class="sub cursor-p pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  4072. <li class="sub cursor-p pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  4073. <li class="sub cursor-p pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  4074. <li class="sub cursor-p pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  4075. </ul>
  4076. </div>
  4077. </div>`);
  4078. element.prepend($button);
  4079. })
  4080. base.waitForKeyElements(config.$baidu.mount.main, (element) => {
  4081. temp.pege = $baidu.detectPage();
  4082. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'youth') return;
  4083. let $button = $(`<div class="wp-s-agile-tool-bar__h-group pl-button">
  4084. <div class="wp-s-agile-tool-bar__h-action is-need-left-sep is-main color-button">
  4085. <button type="button" class="u-button nd-file-list-toolbar-action-item u-button--primary u-button--small is-round is-has-icon pl-button color-button" style="font-size:14px;font-weight:700">
  4086. <i class="u-icon u-icon-more"></i>
  4087. <span>网盘助手</span>
  4088. </button>
  4089. <ul class="dropdown-list nd-common-float-menu pl-dropdown-menu">
  4090. <li class="sub cursor-p pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  4091. <li class="sub cursor-p pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  4092. <li class="sub cursor-p pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  4093. </ul>
  4094. </div>
  4095. </div>`);
  4096. element.prepend($button);
  4097. })
  4098. base.waitForKeyElements(config.$baidu.mount.share, (element) => {
  4099. temp.pege = $baidu.detectPage();
  4100. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  4101. let $button = $(`<a class="g-button tools-share-V20-btn save_btn pl-button color-button" style="padding:0;">
  4102. <span class="g-button-right" style="padding-left:10px">
  4103. <em class="icon icon-download" style="color:#fff;line-height:27px"></em>
  4104. <span class="text" style="width:auto;">下载助手</span>
  4105. </span>
  4106. <ul class="dropdown-list nd-common-float-menu pl-dropdown-menu" style="top:37px">
  4107. <li class="sub cursor-p pl-button-mode pl-button-save"><em class="icon noicon-zhuancun_bai"></em>保存后下载</li>
  4108. <li class="sub cursor-p pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  4109. <li class="sub cursor-p pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  4110. <li class="sub cursor-p pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  4111. </ul>
  4112. </a>`)
  4113. element.after($button);
  4114. })
  4115. },
  4116. addInitButton() {
  4117. base.waitForKeyElements(config.$baidu.mount.home, (element) => {
  4118. temp.pege = $baidu.detectPage();
  4119. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  4120. let $button = $(`<div class="g-dropdown-button pl-button-init" style="opacity:0.5"><div style="color:#fff;" class="g-button g-button-blue color-button"><span class="g-button-right"><em class="icon icon-download" style="color:#fff;"></em><span class="text" style="width:60px;">点我点亮</span></span></div></div>`);
  4121. $button.click(base.showInitDialog);
  4122. element.prepend($button);
  4123. })
  4124. base.waitForKeyElements(config.$baidu.mount.main, (element) => {
  4125. temp.pege = $baidu.detectPage();
  4126. if ($(".pl-button-init").length > 0 || !temp.pege || (temp.pege !== 'main' && temp.pege !== 'youth')) return;
  4127. let $button = $(`<div class="wp-s-agile-tool-bar__h-group pl-button-init">
  4128. <div class="wp-s-agile-tool-bar__h-action is-need-left-sep is-main color-button">
  4129. <button type="button" class="u-button nd-file-list-toolbar-action-item u-button--primary u-button--small is-round is-has-icon pl-button color-button" style="font-size:14px;font-weight:700">
  4130. <i class="u-icon u-icon-download"></i>
  4131. <span>点我点亮</span>
  4132. </button>
  4133. </div>
  4134. </div>`);
  4135. $button.click(base.showInitDialog);
  4136. element.prepend($button);
  4137. })
  4138. base.waitForKeyElements(config.$baidu.mount.share, (element) => {
  4139. temp.pege = $baidu.detectPage();
  4140. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  4141. let $button = $(`<a class="g-button tools-share-V20-btn save_btn pl-button-init color-button" href="javascript:;">
  4142. <span class="g-button-right">
  4143. <em class="icon icon-download" style="color:#fff;line-height:27px"></em>
  4144. <span class="text" style="width:auto;">点我点亮</span>
  4145. </span>
  4146. </a>`)
  4147. $button.click(base.showInitDialog);
  4148. element.after($button);
  4149. })
  4150. },
  4151. async getLink() {
  4152. Swal.fire({
  4153. ...temp.swalDefault,
  4154. showConfirmButton: false,
  4155. allowOutsideClick: false,
  4156. allowEscapeKey: false,
  4157. allowEnterKey: false,
  4158. title: "获取中",
  4159. html: `...`,
  4160. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  4161. customClass: {
  4162. popup: 'loading-popup',
  4163. header: 'loading-header',
  4164. title: 'loading-title',
  4165. content: 'loading-content',
  4166. input: 'loading-input',
  4167. footer: 'loading-footer'
  4168. },
  4169. willOpen: () => {
  4170. Swal.showLoading();
  4171. },
  4172. });
  4173. // 获取选择的文件列表
  4174. let selectList = this.getSelectedList();
  4175. let accessToken = (base.getValue('baidu_access_token') || await $baidu.getToken());
  4176. if (!accessToken) {
  4177. message.info('提示:<br/>稍后请在新标签页中授权助手哦~');
  4178. base.delValue('baidu_access_token');
  4179. await base.sleep(3300);
  4180. GM_openInTab(config.$baidu.api.getAccessToken, { active: true, insert: true, setParent: true })
  4181. let attempts = 0;
  4182. let interval = setInterval(() => {
  4183. if (!!base.getValue('baidu_access_token')) {
  4184. clearInterval(interval);
  4185. accessToken = base.getValue('baidu_access_token')
  4186. }
  4187. attempts++;
  4188. if (attempts > 120) {
  4189. clearInterval(interval);
  4190. return message.error('提示:<br/>时间太长,我先撤下啦~');
  4191. }
  4192. }, 1000);
  4193. return;
  4194. }
  4195. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  4196. if (temp.pege === 'home' || temp.pege === 'main') {
  4197. let cnt = 0;
  4198. let processed = selectList.filter(f => !f.isdir).length;
  4199. async function fetchFiles(dirs) {
  4200. let files = [];
  4201. for (let dir of dirs) {
  4202. $doc.find('.loading-popup .loading-title').html(`文件获取中`);
  4203. let url = `${config.$baidu.api.getFiles}&dir=${encodeURIComponent(dir.path)}&access_token=${accessToken}`;
  4204. let res = await base.get(url, { "User-Agent": config.$baidu.api.ua.downloadLink });
  4205. cnt++;
  4206. if (res?.list?.length && (res.errno === 0 || res.errmsg === "succ")) {
  4207. let subFiles = res.list.filter(f => !f.isdir);
  4208. processed += subFiles.length;
  4209. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} 个文件~</div><div>${dir.path}</div>`);
  4210. files = files.concat(subFiles);
  4211. if (res.list.some(f => f.isdir)) {
  4212. files = files.concat(await fetchFiles(res.list.filter(f => f.isdir)));
  4213. }
  4214. }
  4215. if (cnt >= 50) {
  4216. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} 个文件~</div><div>休息 3 秒...</div>`);
  4217. await base.sleep(3000);
  4218. cnt = 0;
  4219. }
  4220. }
  4221. return files;
  4222. }
  4223. let files = selectList.filter(f => !f.isdir);
  4224. if (selectList.some(f => f.isdir)) {
  4225. files = files.concat(await fetchFiles(selectList.filter(f => f.isdir)));
  4226. }
  4227. if (!files.length) {
  4228. return message.error('提示:<br/>文件夹是空的哦~');
  4229. }
  4230. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  4231. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取文件对应的下载链接~</div>`);
  4232. let fidList = files.map(f => f.fs_id);
  4233. let batchSize = 100;
  4234. let linkList = [];
  4235. for (let i = 0; i < fidList.length; i += batchSize) {
  4236. let url = `${config.$baidu.api.getLink}&fsids=${encodeURIComponent(JSON.stringify(fidList.slice(i, i + batchSize)))}&access_token=${accessToken}`;
  4237. let res = await base.get(url, { "User-Agent": config.$baidu.api.ua.downloadLink });
  4238. if (res.list && res.list.length !== 0 && res.errno === 0) {
  4239. linkList = linkList.concat(res.list);
  4240. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${linkList.length} / ${fidList.length} 个链接~</div>`);
  4241. } else {
  4242. if (res.errno) {
  4243. if (res.errno === 112) {
  4244. return message.error('提示:<br/>页面过期了,刷新重试下吧~<br/>代码:' + res.errno);
  4245. }
  4246. if (res.errno === 9019) {
  4247. base.delValue('baidu_access_token');
  4248. return message.error('提示:<br/>访问令牌已过期,刷新网页后再获取一次吧~<br/>代码:' + res.errno);
  4249. }
  4250. base.delValue('baidu_access_token');
  4251. return message.error('提示:<br/>获取下载链接失败,刷新网页后再试试吧~<br/>代码:' + res.errno);
  4252. } else {
  4253. return message.error('提示:<br/>获取下载链接失败,刷新网页后再试试吧~');
  4254. }
  4255. }
  4256. await base.sleep(1000);
  4257. }
  4258. if (linkList.length) {
  4259. temp.links = [linkList, {
  4260. isFolder: v => v.isdir === 1,
  4261. getFileName: v => (v.server_filename || v.filename),
  4262. getFileSize: v => v.size,
  4263. getFileLink: v => {
  4264. let url = new URL(v.dlink);
  4265. url.searchParams.set("access_token", accessToken);
  4266. return url.href;
  4267. },
  4268. convert: {
  4269. aria2: `--header "User-Agent:${config.$baidu.api.ua.downloadLink}"`,
  4270. curl: `-A "${config.$baidu.api.ua.downloadLink}"`,
  4271. bitcomet: `user_agent=${encodeURIComponent(config.$baidu.api.ua.downloadLink)}`
  4272. },
  4273. tooltip: config.$baidu.dom
  4274. }];
  4275. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  4276. } else {
  4277. return message.error('提示:<br/>获取下载链接失败,刷新网页后再试试吧~');
  4278. }
  4279. } else {
  4280. return message.error('提示:<br/>页面错误~');
  4281. }
  4282. },
  4283. getSelectedList() {
  4284. let List, selectList
  4285. try {
  4286. List = require("system-core:context/context.js").instanceForSystem.list;
  4287. selectList = List.getSelected();
  4288. return selectList;
  4289. } catch (e) { }
  4290. try {
  4291. List = unsafeWindow.document.querySelector('.wp-s-core-pan');
  4292. if (List && List.__vue__.selectedList) {
  4293. selectList = List.__vue__.selectedList;
  4294. return selectList;
  4295. }
  4296. } catch (e) { }
  4297. try {
  4298. List = unsafeWindow.document.querySelector('.file-list');
  4299. if (List && List.__vue__.allFileList) {
  4300. selectList = List.__vue__.allFileList.filter(function (item) { return !!item.selected; });
  4301. return selectList;
  4302. }
  4303. } catch (e) { }
  4304. },
  4305. detectPage() {
  4306. let path = location.pathname;
  4307. if (/^\/disk\/home/.test(path)) return 'home';
  4308. if (/^\/disk\/main/.test(path)) return 'main';
  4309. if (/^\/youth\/pan\/main/.test(path)) return 'youth';
  4310. if (/^\/(s|share)\//.test(path)) return 'share';
  4311. return "";
  4312. },
  4313. async initPanLinker() {
  4314. base.createTip();
  4315. base.registerMenuCommand();
  4316. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  4317. this.addButton();
  4318. } else {
  4319. this.addInitButton();
  4320. }
  4321. this.addPageListener();
  4322. },
  4323. async initAuthorize() {
  4324. base.registerMenuCommand();
  4325. Swal.fire({
  4326. ...temp.swalDefault,
  4327. showConfirmButton: false,
  4328. allowOutsideClick: false,
  4329. allowEscapeKey: false,
  4330. allowEnterKey: false,
  4331. html: `请稍后`,
  4332. willOpen: () => {
  4333. Swal.showLoading();
  4334. },
  4335. });
  4336. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  4337. let url = new URL(location.href);
  4338. let auth = new URL(config.$baidu.api.getAccessToken);
  4339. let allowedClientIds = [
  4340. auth.searchParams.get("client_id"),
  4341. 'L6g70tBRRIXLsY0Z3HwKqlRE', // pcstest_oauth
  4342. 'fSds3K4w43rw37tOqlQmTa2kDwaczK4U', // 小度智能词典笔专业版
  4343. 'TFwtw8uwHxpdkvVqVKdIlx1XqXUnr1zG', // 印象笔记
  4344. '9dgBV9yesuBVOXaxls7aVHbLBLqU8yyg', // WPS文档
  4345. 'l9DdBOG4RYroMscmzK5OChdaGelgd92M', // 小猴云印PC版
  4346. 'Kyr013gHQBf2immy3fQt1jZ3nZVpiGAm', // 简单打印
  4347. 'iYCeC9g08h5vuP9UqvPHKKSVrKFXGa1v', // Alist
  4348. 'omiOnr2tYnN9vSyDErcVFWpPU2mZA7YO', // OpenList
  4349. 'IlLqBbU3GjQ0t46TRwFateTprHWl39zF', // 百度手机助手
  4350. ];
  4351. if (
  4352. /openapi.baidu.com\/oauth\/2.0\/authorize/.test(location.href) &&
  4353. url.searchParams.get("response_type").includes("token") &&
  4354. url.searchParams.get("scope").includes("netdisk") &&
  4355. allowedClientIds.includes(url.searchParams.get("client_id"))
  4356. ) {
  4357. let dialog = await Swal.fire({
  4358. ...temp.swalDefault,
  4359. icon: 'info',
  4360. title: `提示`,
  4361. html: `<p>(◍•ᴗ•◍) 你好呀,为了获取百度网盘文件的下载直链<br/>“下载助手” 需要你的授权,以获取网盘文件的访问令牌</p><br/>
  4362. <p>由于在百度 OAuth 页面使用了其他应用的 Client ID<br/>所以显示的应用名称可能会有所不同,敬请理解</p><br/>
  4363. <p>获取到的令牌仅用于调用百度网盘 API 生成直链<br/>不会用于其他用途,请放心授权</p>`,
  4364. showConfirmButton: true,
  4365. showDenyButton: true,
  4366. allowOutsideClick: false,
  4367. allowEscapeKey: false,
  4368. allowEnterKey: false,
  4369. confirmButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"/></svg> 授权',
  4370. denyButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg> 再想想',
  4371. position: 'center'
  4372. });
  4373. if (dialog.isConfirmed) {
  4374. base.waitForKeyElements("button#auth-allow", function (element) {
  4375. element[0].click();
  4376. }, true)
  4377. return;
  4378. }
  4379. if (dialog.isDenied) {
  4380. return await Swal.fire({
  4381. ...temp.swalDefault,
  4382. icon: 'question',
  4383. title: `好吧(* ̄3 ̄)╭`,
  4384. html: '那就再想一想<br/>想好了就按下 “授权” 按钮吧~',
  4385. timer: 180000,
  4386. toast: true,
  4387. timerProgressBar: true,
  4388. showConfirmButton: false,
  4389. showDenyButton: false,
  4390. position: 'bottom-end',
  4391. })
  4392. }
  4393. } else if (/openapi.baidu.com\/oauth\/2.0\/login_success/.test(location.href)) {
  4394. let int = setInterval(async () => {
  4395. if (location.href.includes('access_token') && (location.href.includes('basic+netdisk') || location.href.includes('basic,netdisk'))) {
  4396. clearInterval(int)
  4397. let token = location.href.match(/access_token=(.*?)&/)[1];
  4398. base.setValue('baidu_access_token', token);
  4399. await Swal.fire({
  4400. ...temp.swalDefault,
  4401. icon: 'success',
  4402. title: `成功啦`,
  4403. html: '<p>(◍•ᴗ•◍) 您已<b>成功授权/授权过</b>脚本获取网盘访问令牌~</p><p>获取到的令牌<b>仅用于调用百度网盘 API 生成直链</b><br/>不会用于其他用途</p><p>等待 <span id="second">/</span> 秒之后将关闭此页面</p>',
  4404. timer: 5000,
  4405. timerProgressBar: true,
  4406. showConfirmButton: true,
  4407. showDenyButton: false,
  4408. allowOutsideClick: false,
  4409. allowEscapeKey: false,
  4410. allowEnterKey: false,
  4411. confirmButtonText: `<svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg> 关闭`,
  4412. willOpen: () => {
  4413. let secondSpan = document.getElementById("second");
  4414. let interval = setInterval(() => {
  4415. if (Swal.isVisible()) {
  4416. let timeLeft = Swal.getTimerLeft();
  4417. if (timeLeft !== null && timeLeft > 0) {
  4418. secondSpan.textContent = (timeLeft / 1000).toFixed(2);
  4419. }
  4420. } else {
  4421. clearInterval(interval);
  4422. }
  4423. }, 10);
  4424. },
  4425. didOpen: function (toast) {
  4426. toast.addEventListener('mouseenter', () => {
  4427. Swal.stopTimer();
  4428. });
  4429. toast.addEventListener('mouseleave', () => {
  4430. Swal.resumeTimer();
  4431. });
  4432. },
  4433. willClose: () => window.close()
  4434. });
  4435. return window.close();
  4436. } else {
  4437. clearInterval(int)
  4438. Swal.close()
  4439. }
  4440. }, 1)
  4441. } else {
  4442. Swal.close()
  4443. }
  4444. } else {
  4445. Swal.close()
  4446. }
  4447. }
  4448. };
  4449. /**
  4450. * 阿里云盘
  4451. * @author 油小猴
  4452. * @author hmjz100
  4453. */
  4454. let $aliyun = {
  4455. addPageListener() {
  4456. $doc.on('click', '.pl-button-save', async function (e) {
  4457. e.preventDefault();
  4458. let reactDomGrid = document.querySelector(config.$aliyun.mount.grid);
  4459. if (reactDomGrid) {
  4460. let dialog = await Swal.fire({
  4461. ...temp.swalDefault,
  4462. title: '提示',
  4463. html: '<div style="display:flex;align-items:center;justify-content:center;">请先切换到&nbsp;&nbsp;<svg class="icon" class="icon--D3kMk " viewBox="0 0 1024 1024" width="20" height="20" fill="currentColor"><use xlink:href="#PDSDrag"></use></svg>&nbsp;<b>列表视图</b>&nbsp;&nbsp;后再获取下载链接哦</div>',
  4464. icon: 'info',
  4465. showCloseButton: true,
  4466. showDenyButton: true,
  4467. confirmButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"/></svg> 切换',
  4468. denyButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg> 不要',
  4469. });
  4470. if (dialog.isConfirmed) {
  4471. document.querySelector(config.$aliyun.mount.switch).click();
  4472. return message.success('提示:<br/>切换为列表视图成功<br/>请再获取一次下载链接吧~');
  4473. }
  4474. return false;
  4475. }
  4476. let selectList = $aliyun.getSelectedList();
  4477. if (selectList.length === 0) {
  4478. return message.error('提示:<br/>请勾选要保存到网盘的文件哦~');
  4479. }
  4480. message.info('提示:<br/>因网盘限制,请保存到自己网盘后再去下载哦~');
  4481. await base.sleep(500);
  4482. document.querySelector('[class*="btn-save--"]').click();
  4483. });
  4484. $doc.on('click', '.pl-button-mode', async function (e) {
  4485. temp.mode = e.currentTarget.dataset.mode;
  4486. if (!temp.mode) return;
  4487. let reactDomGrid = document.querySelector(config.$aliyun.mount.grid);
  4488. if (reactDomGrid) {
  4489. let dialog = await Swal.fire({
  4490. ...temp.swalDefault,
  4491. title: '提示',
  4492. html: '<div style="display:flex;align-items:center;justify-content:center;">请先切换到&nbsp;&nbsp;<svg class="icon" class="icon--D3kMk " viewBox="0 0 1024 1024" width="20" height="20" fill="currentColor"><use xlink:href="#PDSDrag"></use></svg>&nbsp;<b>列表视图</b>&nbsp;&nbsp;后再获取下载链接哦</div>',
  4493. icon: 'info',
  4494. showCloseButton: true,
  4495. showDenyButton: true,
  4496. confirmButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-check"/></svg> 切换',
  4497. denyButtonText: '<svg class="pl-icon"><use xlink:href="#pl-icon-fa-x-mark"/></svg> 不要',
  4498. });
  4499. if (dialog.isConfirmed) {
  4500. document.querySelector(config.$aliyun.mount.switch).click();
  4501. return message.success('提示:<br/>切换为列表视图成功<br/>请再获取一次下载链接吧~');
  4502. }
  4503. return false;
  4504. }
  4505. $aliyun.getLink();
  4506. });
  4507. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  4508. e.preventDefault();
  4509. let o = base._EventFactory(e);
  4510. let $width = o.item.find('.pl-progress-inner');
  4511. let $text = o.item.find('.pl-progress-inner-text');
  4512. let filename = o.link[0].dataset.filename;
  4513. let index = o.link[0].dataset.index;
  4514. let size = Number(o.link[0].dataset.size) || 0;
  4515. let originalHtml = o.link.html();
  4516. base._resetData(index);
  4517. base.get(e.currentTarget.dataset.link, undefined, 'blob', { filename, index });
  4518. let startTime = Date.now();
  4519. let prevLoaded = 0;
  4520. let prevTime = startTime;
  4521. temp.ins[index] = setInterval(async () => {
  4522. let prog = +temp.progress[index] || 0;
  4523. let currentTime = Date.now();
  4524. let elapsedTime = currentTime - startTime;
  4525. let loaded = prog * size / 100;
  4526. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  4527. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  4528. // 计算剩余时间(保护除零)
  4529. let totalProgress = Math.max(prog / 100, 0.01);
  4530. let totalElapsedSeconds = elapsedTime / 1000;
  4531. let estTotalTime = totalElapsedSeconds / totalProgress;
  4532. let remainingTime = estTotalTime - totalElapsedSeconds;
  4533. // 更新界面状态
  4534. o.link.hide();
  4535. o.directLink.hide();
  4536. o.tip.hide();
  4537. o.stop.show();
  4538. o.copy.hide();
  4539. o.progress.show();
  4540. // 更新进度条
  4541. $width.css('width', `${prog}%`);
  4542. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  4543. // 更新历史值
  4544. prevLoaded = loaded;
  4545. prevTime = currentTime;
  4546. // 下载完成
  4547. if (prog >= 100) {
  4548. await base.sleep(1000);
  4549. clearInterval(temp.ins[index]);
  4550. temp.progress[index] = 0;
  4551. o.item.find('.pl-progress-stop').hide();
  4552. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  4553. o.back.show();
  4554. await base.sleep(3000);
  4555. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  4556. }
  4557. }, 500);
  4558. });
  4559. $doc.on('click', '.listener-aria2-download', async function (e) {
  4560. let target = $(e.currentTarget);
  4561. if (target.attr('data-processing') === 'true') return;
  4562. target.attr('data-processing', 'true');
  4563. let originalHtml = target.html();
  4564. target.find(".pl-icon").remove();
  4565. target.find('.pl-loading').remove();
  4566. target.prepend(base.createLoading());
  4567. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"), [`Referer:https://${location.host}/`]);
  4568. if (res === 'success') {
  4569. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  4570. } else {
  4571. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  4572. }
  4573. await base.sleep(3000);
  4574. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  4575. });
  4576. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  4577. let target = $(e.currentTarget);
  4578. if (target.attr('data-processing') === 'true') return;
  4579. target.attr('data-processing', 'true');
  4580. let originalHtml = target.html();
  4581. target.find(".pl-icon").remove();
  4582. target.find('.pl-loading').remove();
  4583. target.prepend(base.createLoading());
  4584. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"), { "referrer": `https://${location.host}/` });
  4585. if (res === 'success') {
  4586. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  4587. } else {
  4588. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  4589. }
  4590. await base.sleep(3000);
  4591. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  4592. });
  4593. $doc.on('click', '.listener-abdm-download', async function (e) {
  4594. let target = $(e.currentTarget);
  4595. if (target.attr('data-processing') === 'true') return;
  4596. target.attr('data-processing', 'true');
  4597. let originalHtml = target.html();
  4598. target.find(".pl-icon").remove();
  4599. target.find('.pl-loading').remove();
  4600. target.prepend(base.createLoading());
  4601. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), undefined);
  4602. if (res === 'success') {
  4603. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  4604. } else {
  4605. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  4606. }
  4607. await base.sleep(3000);
  4608. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  4609. });
  4610. },
  4611. greenerPage() {
  4612. base.waitForKeyElements('[class*="share-list-banner"]', function (tag) {
  4613. tag.fadeOut();
  4614. }, true);
  4615. base.waitForKeyElements('[class*="to-app"]', function (tag) {
  4616. tag.fadeOut();
  4617. }, true);
  4618. base.waitForKeyElements('[class*="btn-mobile-save"]', function (tag) {
  4619. tag.fadeOut();
  4620. }, true);
  4621. base.waitForKeyElements('[class*="SplashScreenImg--close"]', function (tag) {
  4622. tag[0].click();
  4623. }, true);
  4624. base.waitForKeyElements('[class*="container"]', function (tag) {
  4625. tag.find('[class^="icon-close"]').click();
  4626. }, true);
  4627. base.waitForKeyElements('[class*="popup_main_close"]', function (tag) {
  4628. tag[0].click();
  4629. }, true);
  4630. },
  4631. beautifyPage() {
  4632. base.adaptiveThemeOverride([
  4633. ['#3763ff', temp.color],
  4634. ['#8664ff', `${temp.color}D0`],
  4635. ['99, 125, 255', base.hexToRgba(temp.color)],
  4636. ['132, 133, 141', base.hexToRgba(temp.color)],
  4637. ['112, 136, 255', base.hexToRgba(temp.color)],
  4638. ['97, 122, 250', base.hexToRgba(temp.color)],
  4639. ['68, 109, 255', base.hexToRgba(temp.color)],
  4640. ['82, 110, 250', base.hexToRgba(`${temp.color}20`)],
  4641. ['122, 144, 255', base.hexToRgba(`${temp.color}D0`)],
  4642. ['138, 157, 255', base.hexToRgba(`${temp.color}D0`)],
  4643. ]);
  4644. },
  4645. svg: `<svg class="ali-btn-icon" style="margin-right:3px;" 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>`,
  4646. addButton() {
  4647. base.waitForKeyElements(config.$aliyun.mount.home, (element) => {
  4648. temp.pege = $aliyun.detectPage();
  4649. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  4650. let $button = $(`<div class="ali-button pl-button">
  4651. <span data-role="icon" data-render-as="svg" class="icon">${$aliyun.svg}下载助手</span>
  4652. <ul class="pl-dropdown-menu" style="top:30px; right:0;">
  4653. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  4654. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  4655. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  4656. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  4657. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  4658. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  4659. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  4660. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  4661. </ul>
  4662. </div>`);
  4663. element.append($button);
  4664. })
  4665. base.waitForKeyElements(config.$aliyun.mount.share, (element) => {
  4666. temp.pege = $aliyun.detectPage();
  4667. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  4668. let $button = $(`<div class="ali-button pl-button">
  4669. <span data-role="icon" data-render-as="svg" class="icon">${$aliyun.svg}下载助手</span>
  4670. <ul class="pl-dropdown-menu" style="top:30px; right:16px;">
  4671. <li class="pl-button-mode pl-button-save"><use xlink:href="#pl-icon-fa-save"/></svg>保存后下载</li>
  4672. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  4673. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  4674. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  4675. </ul>
  4676. </div>`);
  4677. $button.css({ 'margin-right': '10px', "height": "36px", "width": "auto", "padding": "1px 30px" });
  4678. element.prepend($button);
  4679. })
  4680. },
  4681. addInitButton() {
  4682. let $button = $(`<div class="ali-button pl-button-init"><span data-role="icon" data-render-as="svg" class="icon">${$aliyun.svg}点我点亮</span></div>`);
  4683. $button.click(base.showInitDialog);
  4684. base.waitForKeyElements(config.$aliyun.mount.home, (element) => {
  4685. temp.pege = $aliyun.detectPage();
  4686. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  4687. $button.css({ "width": "auto" });
  4688. element.append($button);
  4689. })
  4690. base.waitForKeyElements(config.$aliyun.mount.share, (element) => {
  4691. temp.pege = $aliyun.detectPage();
  4692. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  4693. $button.css({ 'margin-right': '10px', "height": "36px", "padding": "1px 30px", "width": "auto" });
  4694. element.prepend($button);
  4695. })
  4696. },
  4697. async getLink() {
  4698. Swal.fire({
  4699. ...temp.swalDefault,
  4700. showConfirmButton: false,
  4701. allowOutsideClick: false,
  4702. allowEscapeKey: false,
  4703. allowEnterKey: false,
  4704. title: "获取中",
  4705. html: `...`,
  4706. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  4707. customClass: {
  4708. popup: 'loading-popup',
  4709. header: 'loading-header',
  4710. title: 'loading-title',
  4711. content: 'loading-content',
  4712. input: 'loading-input',
  4713. footer: 'loading-footer'
  4714. },
  4715. willOpen: () => {
  4716. Swal.showLoading();
  4717. },
  4718. });
  4719. let selectList = this.getSelectedList();
  4720. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  4721. if (selectList.every(item => item.type !== 'file')) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  4722. if (temp.pege === 'home') {
  4723. selectList = selectList.filter(item => item.type === 'file')
  4724. let batchSize = 15;
  4725. let processed = 0;
  4726. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  4727. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取文件对应的下载链接~</div>`);
  4728. for (let i = 0; i < selectList.length; i += batchSize) {
  4729. // 当前批次文件
  4730. let batch = selectList.slice(i, i + batchSize);
  4731. // 过滤掉已有 URL 的文件
  4732. let noUrlSelectList = batch.filter(v => !Boolean(v.url));
  4733. let hasUrlSelectList = batch.filter(v => Boolean(v.url));
  4734. let queue = [];
  4735. // 为没有 URL 的文件生成请求队列
  4736. noUrlSelectList.forEach((item) => {
  4737. queue.push(this.getFileUrlByOnce(item.driveId, item.fileId)
  4738. .then(val => {
  4739. processed++;
  4740. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  4741. return val;
  4742. }));
  4743. });
  4744. hasUrlSelectList.forEach((item) => {
  4745. processed++;
  4746. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  4747. });
  4748. // 等待本批次的请求结果
  4749. let res = await Promise.all(queue);
  4750. res.forEach((val, index) => {
  4751. noUrlSelectList[index].url = val;
  4752. });
  4753. // 每次处理完一个批次后,等待 1 秒
  4754. await base.sleep(1000);
  4755. }
  4756. } else {
  4757. return message.error('提示:<br/>页面错误~');
  4758. }
  4759. temp.links = [selectList, {
  4760. isFolder: v => v.type === 'folder',
  4761. getFileName: v => v.name,
  4762. getFileSize: v => v.size,
  4763. getFileLink: v => (v.downloadUrl || v.url),
  4764. convert: {
  4765. aria2: `--header "Referer:https://${location.host}/"`,
  4766. curl: `-e "https://${location.host}/"`,
  4767. bitcomet: `&refer=${encodeURIComponent(`https://${location.host}/`)}`
  4768. },
  4769. tooltip: config.$aliyun.dom
  4770. }];
  4771. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  4772. },
  4773. async getFileUrlByOnce(d, f) {
  4774. let res = await base.post(config.$aliyun.api.getLink, { drive_id: d, file_id: f }, { "Authorization": `${base.getStorage('token').token_type} ${base.getStorage('token').access_token}`, "X-Canary": "client=windows,app=adrive,version=v6.0.0" });
  4775. if (res.code === 'AccessTokenInvalid') {
  4776. return message.error('提示:<br/>访问令牌过期了,请刷新网页后再试');
  4777. }
  4778. if (res.url) {
  4779. return res.url;
  4780. }
  4781. return "";
  4782. },
  4783. getSelectedList() {
  4784. try {
  4785. let selectedList = [];
  4786. let reactDom = document.querySelector(config.$aliyun.mount.list);
  4787. let reactObj = base.findReact(reactDom, 1);
  4788. let props = reactObj.pendingProps;
  4789. if (props) {
  4790. let fileList = props.dataSource || [];
  4791. let selectedKeys = props.selectedKeys.split(',');
  4792. fileList.forEach(function (val) {
  4793. if (selectedKeys.includes(val.fileId)) {
  4794. selectedList.push(val);
  4795. }
  4796. });
  4797. }
  4798. return selectedList;
  4799. } catch (e) {
  4800. return [];
  4801. }
  4802. },
  4803. detectPage() {
  4804. let path = location.pathname;
  4805. if (/^\/(drive)/.test(path)) return 'home';
  4806. if (/^\/(s|share)\//.test(path)) return 'share';
  4807. return "";
  4808. },
  4809. async initPanLinker() {
  4810. base.createTip();
  4811. base.registerMenuCommand();
  4812. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  4813. this.addButton();
  4814. } else {
  4815. this.addInitButton();
  4816. }
  4817. this.addPageListener();
  4818. },
  4819. };
  4820. /**
  4821. * 中国移动云盘 / 和彩云
  4822. * @author 油小猴
  4823. * @author hmjz100
  4824. */
  4825. let $mcloud = {
  4826. addPageListener() {
  4827. $doc.on('click', '.pl-button-mode', async function (e) {
  4828. temp.mode = e.currentTarget.dataset.mode;
  4829. if (!temp.mode) return;
  4830. $mcloud.getLink();
  4831. });
  4832. $doc.on('click', '.pl-button-save', async function (e) {
  4833. e.preventDefault();
  4834. let selectList = $mcloud.getSelectedList();
  4835. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  4836. if (selectList.every(item => !item.contentID && !item.contentName)) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  4837. message.info('提示:<br/>因网盘限制,只能够通过页面直接下载哦~');
  4838. await base.sleep(500);
  4839. document.querySelector('.btn-top.btn-top_dl').click();
  4840. });
  4841. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  4842. e.preventDefault();
  4843. let o = base._EventFactory(e);
  4844. let $width = o.item.find('.pl-progress-inner');
  4845. let $text = o.item.find('.pl-progress-inner-text');
  4846. let filename = o.link[0].dataset.filename;
  4847. let index = o.link[0].dataset.index;
  4848. let size = Number(o.link[0].dataset.size) || 0;
  4849. let originalHtml = o.link.html();
  4850. base._resetData(index);
  4851. base.get(e.currentTarget.dataset.link, undefined, 'blob', { filename, index });
  4852. let startTime = Date.now();
  4853. let prevLoaded = 0;
  4854. let prevTime = startTime;
  4855. temp.ins[index] = setInterval(async () => {
  4856. let prog = +temp.progress[index] || 0;
  4857. let currentTime = Date.now();
  4858. let elapsedTime = currentTime - startTime;
  4859. let loaded = prog * size / 100;
  4860. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  4861. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  4862. // 计算剩余时间(保护除零)
  4863. let totalProgress = Math.max(prog / 100, 0.01);
  4864. let totalElapsedSeconds = elapsedTime / 1000;
  4865. let estTotalTime = totalElapsedSeconds / totalProgress;
  4866. let remainingTime = estTotalTime - totalElapsedSeconds;
  4867. // 更新界面状态
  4868. o.link.hide();
  4869. o.directLink.hide();
  4870. o.tip.hide();
  4871. o.stop.show();
  4872. o.copy.hide();
  4873. o.progress.show();
  4874. // 更新进度条
  4875. $width.css('width', `${prog}%`);
  4876. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  4877. // 更新历史值
  4878. prevLoaded = loaded;
  4879. prevTime = currentTime;
  4880. // 下载完成
  4881. if (prog >= 100) {
  4882. await base.sleep(1000);
  4883. clearInterval(temp.ins[index]);
  4884. temp.progress[index] = 0;
  4885. o.item.find('.pl-progress-stop').hide();
  4886. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  4887. o.back.show();
  4888. await base.sleep(3000);
  4889. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  4890. }
  4891. }, 500);
  4892. });
  4893. $doc.on('click', '.listener-aria2-download', async function (e) {
  4894. let target = $(e.currentTarget);
  4895. if (target.attr('data-processing') === 'true') return;
  4896. target.attr('data-processing', 'true');
  4897. let originalHtml = target.html();
  4898. target.find(".pl-icon").remove();
  4899. target.find('.pl-loading').remove();
  4900. target.prepend(base.createLoading());
  4901. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"));
  4902. if (res === 'success') {
  4903. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  4904. } else {
  4905. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  4906. }
  4907. await base.sleep(3000);
  4908. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  4909. });
  4910. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  4911. let target = $(e.currentTarget);
  4912. if (target.attr('data-processing') === 'true') return;
  4913. target.attr('data-processing', 'true');
  4914. let originalHtml = target.html();
  4915. target.find(".pl-icon").remove();
  4916. target.find('.pl-loading').remove();
  4917. target.prepend(base.createLoading());
  4918. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"));
  4919. if (res === 'success') {
  4920. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  4921. } else {
  4922. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  4923. }
  4924. await base.sleep(3000);
  4925. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  4926. });
  4927. $doc.on('click', '.listener-abdm-download', async function (e) {
  4928. let target = $(e.currentTarget);
  4929. if (target.attr('data-processing') === 'true') return;
  4930. target.attr('data-processing', 'true');
  4931. let originalHtml = target.html();
  4932. target.find(".pl-icon").remove();
  4933. target.find('.pl-loading').remove();
  4934. target.prepend(base.createLoading());
  4935. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), undefined);
  4936. if (res === 'success') {
  4937. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  4938. } else {
  4939. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  4940. }
  4941. await base.sleep(3000);
  4942. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  4943. });
  4944. },
  4945. greenerPage() {
  4946. base.waitForKeyElements(".adv_swiper_menu", function (tag) {
  4947. tag.fadeOut();
  4948. }, true);
  4949. base.waitForKeyElements(".client-bubble", function (tag) {
  4950. tag.fadeOut();
  4951. }, true);
  4952. base.waitForKeyElements(".avs-box", function (tag) {
  4953. tag.fadeOut();
  4954. }, true);
  4955. base.waitForKeyElements(".top-adv-swiper", function (tag) {
  4956. tag.fadeOut();
  4957. }, true);
  4958. base.waitForKeyElements(".client_download_icon", function (tag) {
  4959. tag.fadeOut();
  4960. }, true);
  4961. base.waitForKeyElements(".document_top_memberCenter", function (tag) {
  4962. $(tag[0]).click(function () {
  4963. Swal.fire({
  4964. ...temp.swalDefault,
  4965. html: `<iframe style="height:700px; width:420px; border:0;" src="https://vip.yun.139.com/vip/"></iframe>`,
  4966. allowOutsideClick: false,
  4967. showCloseButton: true,
  4968. showConfirmButton: false,
  4969. });
  4970. });
  4971. }, true);
  4972. },
  4973. beautifyPage() {
  4974. base.adaptiveThemeOverride([
  4975. ['#3181f9', temp.color],
  4976. ['#5a9afa', temp.color],
  4977. ['#98c0fc', `${temp.color}D0`],
  4978. ['#2d76e5', `${temp.color}D0`],
  4979. ['49,129,249,.08', base.hexToRgba(`${temp.color}20`)],
  4980. ]);
  4981. },
  4982. addButton() {
  4983. base.waitForKeyElements(config.$mcloud.mount.home, (element) => {
  4984. temp.pege = $mcloud.detectPage();
  4985. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  4986. let $button = $(`<div class="pl-button mcloud-button btn-top">
  4987. <span class="mcloud-btn">下载助手</span>
  4988. <ul class="pl-dropdown-menu" style="top:36px; left:0; letter-spacing:normal;">
  4989. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  4990. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  4991. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  4992. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  4993. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  4994. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  4995. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  4996. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  4997. </ul>
  4998. </div>`);
  4999. element.prepend($button);
  5000. })
  5001. base.waitForKeyElements(config.$mcloud.mount.share, (element) => {
  5002. temp.pege = $mcloud.detectPage();
  5003. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  5004. let $button = $(`<div class="pl-button mcloud-share-button">
  5005. <span class="mcloud-btn">下载助手</span>
  5006. <ul class="pl-dropdown-menu" style="top:36px; left:0; letter-spacing:normal;">
  5007. <li class="pl-button-mode pl-button-save"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>直接下载</li>
  5008. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  5009. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  5010. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  5011. </ul>
  5012. </div>`);
  5013. element.prepend($button);
  5014. })
  5015. },
  5016. addInitButton() {
  5017. let $button = $(`<div class="pl-button-init"><span class="mcloud-btn">点我点亮</span></div>`);
  5018. $button.click(base.showInitDialog);
  5019. base.waitForKeyElements(config.$mcloud.mount.home, (element) => {
  5020. temp.pege = $mcloud.detectPage();
  5021. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  5022. $button.addClass('mcloud-button');
  5023. element.prepend($button);
  5024. })
  5025. base.waitForKeyElements(config.$mcloud.mount.share, (element) => {
  5026. temp.pege = $mcloud.detectPage();
  5027. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  5028. $button.addClass('mcloud-share-button').css({ "cursor": "pointer" });
  5029. element.prepend($button);
  5030. })
  5031. },
  5032. getRandomString(len) {
  5033. len = len || 16;
  5034. let $chars = 'ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678';
  5035. let maxPos = $chars.length;
  5036. let pwd = "";
  5037. for (let i = 0; i < len; i++) {
  5038. pwd += $chars.charAt(Math.floor(Math.random() * maxPos));
  5039. }
  5040. return pwd;
  5041. },
  5042. utob(str) {
  5043. let u = String.fromCharCode;
  5044. return str.replace(/[\uD800-\uDBFF][\uDC00-\uDFFFF]|[^\x00-\x7F]/g, function (t) {
  5045. if (t.length < 2) {
  5046. let e = t.charCodeAt(0);
  5047. return e < 128 ? t : e < 2048 ? u(192 | e >>> 6) + u(128 | 63 & e) : u(224 | e >>> 12 & 15) + u(128 | e >>> 6 & 63) + u(128 | 63 & e);
  5048. }
  5049. e = 65536 + 1024 * (t.charCodeAt(0) - 55296) + (t.charCodeAt(1) - 56320);
  5050. return u(240 | e >>> 18 & 7) + u(128 | e >>> 12 & 63) + u(128 | e >>> 6 & 63) + u(128 | 63 & e);
  5051. });
  5052. },
  5053. getSign(e, t, a, n) {
  5054. let r = "",
  5055. i = "";
  5056. if (t) {
  5057. let s = Object.assign({}, t);
  5058. i = JSON.stringify(s),
  5059. i = i.replace(/\s*/g, ""),
  5060. i = encodeURIComponent(i);
  5061. let c = i.split(""),
  5062. u = c.sort();
  5063. i = u.join("");
  5064. }
  5065. let A = md5(base.encodeBase(this.utob(i)));
  5066. let l = md5(a + ":" + n);
  5067. return md5(A + l).toUpperCase();
  5068. },
  5069. async getFileUrlByOnce(item, index) {
  5070. try {
  5071. if (item.downloadUrl) return {
  5072. index,
  5073. downloadUrl: item.downloadUrl
  5074. };
  5075. if (this.detectPage() === 'home') {
  5076. let body = {
  5077. fileId: item.contentID
  5078. }
  5079. let time = new Date(+new Date() + 8 * 3600 * 1000).toJSON().substr(0, 19).replace('T', ' ');
  5080. let key = this.getRandomString(16);
  5081. let sign = this.getSign(undefined, body, time, key);
  5082. let getCookie = (name) => {
  5083. let cname = name + "=";
  5084. let ca = document.cookie.split(';');
  5085. for (let i = 0; i < ca.length; i++) {
  5086. let c = ca[i].trim();
  5087. if (c.indexOf(cname) == 0) return c.substring(cname.length, c.length);
  5088. }
  5089. return "";
  5090. }
  5091. let res = await base.post(config.$mcloud.api.getLink, body, {
  5092. "Authorization": getCookie("authorization"),
  5093. "Caller": "web",
  5094. "CMS-DEVICE": "default",
  5095. "Mcloud-Channel": "1000101",
  5096. "Mcloud-Client": "10701",
  5097. "Mcloud-Sign": time + "," + key + "," + sign,
  5098. "Mcloud-Version": "7.14.2",
  5099. "X-DeviceInfo": "||9|7.14.2|chrome|119.0.0.0|||windows 10||zh-CN|||",
  5100. "X-Huawei-ChannelSrc": "10000034",
  5101. "X-Inner-Ntwk": "2",
  5102. "X-M4C-Caller": "PC",
  5103. "X-M4C-Src": "10002",
  5104. "X-SvcType": "1",
  5105. "X-Yun-Api-Version": "v1",
  5106. "X-Yun-App-Channel": "10000034",
  5107. "X-Yun-Channel-Source": "10000034",
  5108. "X-Yun-Client-Info": "||9|7.14.2|chrome|119.0.0.0|||windows 10||zh-CN|||||",
  5109. "X-Yun-Module-Type": "100",
  5110. "X-Yun-Svc-Type": "1"
  5111. });
  5112. if (res.success) {
  5113. return {
  5114. index,
  5115. downloadUrl: res.data.url
  5116. };
  5117. } else {
  5118. return {
  5119. index,
  5120. downloadUrl: '获取下载地址失败,刷新后再试试吧~'
  5121. };
  5122. }
  5123. }
  5124. if (this.detectPage() === 'share') {
  5125. let vueDom = document.querySelector(".main_file_list").__vue__;
  5126. let res = await base.post(config.$mcloud.api.getShareLink, `linkId=${vueDom.linkID}&contentIds=${item.path}&catalogIds=`, { "Content-Type": "application/x-www-form-urlencoded" });
  5127. if (res.code === 0) {
  5128. return {
  5129. index,
  5130. downloadUrl: res.data.redrUrl
  5131. };
  5132. } else {
  5133. return {
  5134. index,
  5135. downloadUrl: '获取下载地址失败,刷新后再试试吧~'
  5136. };
  5137. }
  5138. }
  5139. } catch (e) {
  5140. return {
  5141. index,
  5142. downloadUrl: '获取下载地址失败,刷新后再试试吧~'
  5143. };
  5144. }
  5145. },
  5146. async getLink() {
  5147. Swal.fire({
  5148. ...temp.swalDefault,
  5149. showConfirmButton: false,
  5150. allowOutsideClick: false,
  5151. allowEscapeKey: false,
  5152. allowEnterKey: false,
  5153. title: "获取中",
  5154. html: `...`,
  5155. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  5156. customClass: {
  5157. popup: 'loading-popup',
  5158. header: 'loading-header',
  5159. title: 'loading-title',
  5160. content: 'loading-content',
  5161. input: 'loading-input',
  5162. footer: 'loading-footer'
  5163. },
  5164. willOpen: function () {
  5165. Swal.showLoading();
  5166. },
  5167. });
  5168. let selectList = this.getSelectedList();
  5169. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  5170. if (selectList.every(item => !item.contentID && !item.contentName)) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  5171. if (temp.pege === 'home') {
  5172. selectList = selectList.filter(item => item.contentID && item.contentName && item.contentSuffix);
  5173. let batchSize = 15;
  5174. let processed = 0;
  5175. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  5176. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取文件对应的下载链接~</div>`);
  5177. for (let i = 0; i < selectList.length; i += batchSize) {
  5178. let batch = selectList.slice(i, i + batchSize);
  5179. let queue = [];
  5180. batch.forEach((item, localIndex) => {
  5181. let globalIndex = i + localIndex;
  5182. queue.push(this.getFileUrlByOnce(item, globalIndex)
  5183. .then(val => {
  5184. processed++;
  5185. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  5186. return val;
  5187. }));
  5188. });
  5189. let res = await Promise.all(queue);
  5190. res.forEach(val => {
  5191. selectList[val.index].downloadUrl = val.downloadUrl;
  5192. });
  5193. await base.sleep(1000);
  5194. }
  5195. } else {
  5196. return message.error('提示:<br/>页面错误~');
  5197. }
  5198. temp.links = [selectList, {
  5199. isFolder: v => (v.dirEtag || v.caName),
  5200. getFileName: v => (v.contentName || v.coName),
  5201. getFileSize: v => (v.contentSize || v.coSize),
  5202. getFileLink: v => v.downloadUrl,
  5203. tooltip: config.$mcloud.dom
  5204. }];
  5205. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  5206. },
  5207. getSelectedList() {
  5208. try {
  5209. return document.querySelector(".main_file_list").__vue__.selectList.map(val => val.item);
  5210. } catch (e) {
  5211. let vueDom = document.querySelector(".home-page").__vue__;
  5212. let fileList = vueDom._computedWatchers.fileList.value;
  5213. let dirList = vueDom._computedWatchers.dirList.value;
  5214. let selectedFileIndex = vueDom.selectedFile;
  5215. let selectedDirIndex = vueDom.selectedDir;
  5216. let selectFileList = fileList.filter((v, i) => {
  5217. return selectedFileIndex.includes(i);
  5218. });
  5219. let selectDirList = dirList.filter((v, i) => {
  5220. return selectedDirIndex.includes(i);
  5221. });
  5222. return [...selectFileList, ...selectDirList];
  5223. }
  5224. },
  5225. detectPage() {
  5226. let path = location.pathname;
  5227. if (/^\/w/.test(path)) return 'home';
  5228. if (/^\/link|shareweb/.test(path)) return 'share';
  5229. return "";
  5230. },
  5231. async initPanLinker() {
  5232. base.createTip();
  5233. base.registerMenuCommand();
  5234. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  5235. this.addButton();
  5236. } else {
  5237. this.addInitButton();
  5238. }
  5239. this.addPageListener();
  5240. },
  5241. };
  5242. /**
  5243. * 天翼云盘
  5244. * @author 油小猴
  5245. * @author hmjz100
  5246. */
  5247. let $tcloud = {
  5248. addPageListener() {
  5249. $doc.on('click', '.pl-button-mode', async function (e) {
  5250. temp.mode = e.currentTarget.dataset.mode;
  5251. if (!temp.mode) return;
  5252. $tcloud.getLink();
  5253. });
  5254. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  5255. e.preventDefault();
  5256. let o = base._EventFactory(e);
  5257. let $width = o.item.find('.pl-progress-inner');
  5258. let $text = o.item.find('.pl-progress-inner-text');
  5259. let filename = o.link[0].dataset.filename;
  5260. let index = o.link[0].dataset.index;
  5261. let size = Number(o.link[0].dataset.size) || 0;
  5262. let originalHtml = o.link.html();
  5263. base._resetData(index);
  5264. base.get(e.currentTarget.dataset.link, undefined, 'blob', { filename, index });
  5265. let startTime = Date.now();
  5266. let prevLoaded = 0;
  5267. let prevTime = startTime;
  5268. temp.ins[index] = setInterval(async function () {
  5269. let prog = +temp.progress[index] || 0;
  5270. let currentTime = Date.now();
  5271. let elapsedTime = currentTime - startTime;
  5272. let loaded = prog * size / 100;
  5273. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  5274. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  5275. // 计算剩余时间(保护除零)
  5276. let totalProgress = Math.max(prog / 100, 0.01);
  5277. let totalElapsedSeconds = elapsedTime / 1000;
  5278. let estTotalTime = totalElapsedSeconds / totalProgress;
  5279. let remainingTime = estTotalTime - totalElapsedSeconds;
  5280. // 更新界面状态
  5281. o.link.hide();
  5282. o.directLink.hide();
  5283. o.tip.hide();
  5284. o.stop.show();
  5285. o.copy.hide();
  5286. o.progress.show();
  5287. // 更新进度条
  5288. $width.css('width', `${prog}%`);
  5289. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  5290. // 更新历史值
  5291. prevLoaded = loaded;
  5292. prevTime = currentTime;
  5293. // 下载完成
  5294. if (prog >= 100) {
  5295. await base.sleep(1000);
  5296. clearInterval(temp.ins[index]);
  5297. temp.progress[index] = 0;
  5298. o.item.find('.pl-progress-stop').hide();
  5299. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  5300. o.back.show();
  5301. await base.sleep(3000);
  5302. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  5303. }
  5304. }, 500);
  5305. });
  5306. $doc.on('click', '.listener-aria2-download', async function (e) {
  5307. let target = $(e.currentTarget);
  5308. if (target.attr('data-processing') === 'true') return;
  5309. target.attr('data-processing', 'true');
  5310. let originalHtml = target.html();
  5311. target.find(".pl-icon").remove();
  5312. target.find('.pl-loading').remove();
  5313. target.prepend(base.createLoading());
  5314. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"));
  5315. if (res === 'success') {
  5316. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  5317. } else {
  5318. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  5319. }
  5320. await base.sleep(3000);
  5321. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  5322. });
  5323. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  5324. let target = $(e.currentTarget);
  5325. if (target.attr('data-processing') === 'true') return;
  5326. target.attr('data-processing', 'true');
  5327. let originalHtml = target.html();
  5328. target.find(".pl-icon").remove();
  5329. target.find('.pl-loading').remove();
  5330. target.prepend(base.createLoading());
  5331. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"));
  5332. if (res === 'success') {
  5333. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  5334. } else {
  5335. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  5336. }
  5337. await base.sleep(3000);
  5338. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  5339. });
  5340. $doc.on('click', '.listener-abdm-download', async function (e) {
  5341. let target = $(e.currentTarget);
  5342. if (target.attr('data-processing') === 'true') return;
  5343. target.attr('data-processing', 'true');
  5344. let originalHtml = target.html();
  5345. target.find(".pl-icon").remove();
  5346. target.find('.pl-loading').remove();
  5347. target.prepend(base.createLoading());
  5348. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), undefined);
  5349. if (res === 'success') {
  5350. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  5351. } else {
  5352. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  5353. }
  5354. await base.sleep(3000);
  5355. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  5356. });
  5357. },
  5358. greenerPage() {
  5359. base.waitForKeyElements(".advertising-mask", function (tag) {
  5360. tag.fadeOut();
  5361. }, true);
  5362. base.waitForKeyElements("a.client-download.nav-block", function (tag) {
  5363. tag.fadeOut();
  5364. }, true);
  5365. },
  5366. beautifyPage() {
  5367. base.adaptiveThemeOverride([
  5368. ['#2b89ea', temp.color],
  5369. ['#1874d3', `${temp.color}F0`],
  5370. ['#1890ff', temp.color],
  5371. ['#388fc9', temp.color],
  5372. ['#0087ff', temp.color],
  5373. ['#255697', temp.color],
  5374. ['#3ea6ff', `${temp.color}80`],
  5375. ['#1d52f2', temp.color],
  5376. ['#3699ff', `${temp.color}D0`],
  5377. ['#f4f9fe', `${temp.color}10`],
  5378. ['#eaf5ff', `${temp.color}20`],
  5379. ], "other");
  5380. },
  5381. addButton() {
  5382. let $button = $(`<div class="pl-button tcloud-button">
  5383. 下载助手&nbsp;
  5384. <i aria-label="icon:caret-down" class="anticon anticon-caret-down">
  5385. <svg viewBox="0 0 1024 1024" data-icon="caret-down" width="1em" height="1em" fill="currentColor" aria-hidden="true" focusable="false">
  5386. <path d="M840.4 300H183.6c-19.7 0-30.7 20.8-18.5 35l328.4 380.8c9.4 10.9 27.5 10.9 37 0L858.9 335c12.2-14.2 1.2-35-18.5-35z"></path>
  5387. </svg>
  5388. </i>
  5389. <ul class="pl-dropdown-menu">
  5390. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  5391. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  5392. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  5393. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  5394. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  5395. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  5396. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  5397. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  5398. </ul>
  5399. </div>`);
  5400. $button.find(".pl-dropdown-menu").css({ 'position': 'absolute', 'left': '-1px' })
  5401. base.waitForKeyElements(config.$tcloud.mount.home, (element) => {
  5402. temp.pege = $tcloud.detectPage();
  5403. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  5404. $button.find(".pl-dropdown-menu").css({ 'top': '28px' })
  5405. element.prepend($button);
  5406. })
  5407. base.waitForKeyElements(config.$tcloud.mount.share, (element) => {
  5408. temp.pege = $tcloud.detectPage();
  5409. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  5410. $button.css({ 'height': '28px', 'border-radius': '15px' })
  5411. $button.find(".pl-dropdown-menu").css({ 'top': '25px' })
  5412. element.prepend($button);
  5413. })
  5414. },
  5415. addInitButton() {
  5416. let $button = $(`<div class="tcloud-button pl-button-init">点我点亮</div>`);
  5417. $button.click(base.showInitDialog);
  5418. base.waitForKeyElements(config.$tcloud.mount.home, (element) => {
  5419. temp.pege = $tcloud.detectPage();
  5420. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  5421. element.prepend($button);
  5422. })
  5423. base.waitForKeyElements(config.$tcloud.mount.share, (element) => {
  5424. temp.pege = $tcloud.detectPage();
  5425. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  5426. $button.css({ 'height': '28px', 'border-radius': '15px' })
  5427. element.prepend($button);
  5428. })
  5429. },
  5430. async getToken() {
  5431. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  5432. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取状态~</div>`);
  5433. let res = await base.getFinalUrl(config.$tcloud.api.getAccessToken, undefined, true);
  5434. let accessToken = res.match(/accessToken=(\w+)/)?.[1];
  5435. accessToken && base.setStorage('accessToken', accessToken);
  5436. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  5437. $doc.find('.loading-popup .swal2-html-container').html(`<div>获取成功,令牌已缓存~</div>`);
  5438. return accessToken;
  5439. },
  5440. async getFileUrlByOnce(item, index, token) {
  5441. try {
  5442. if (item.downloadUrl) {
  5443. return {
  5444. index,
  5445. downloadUrl: item.downloadUrl
  5446. }
  5447. };
  5448. let time = Date.now(),
  5449. fileId = item.fileId,
  5450. o = "AccessToken=" + token + "&Timestamp=" + time + "&fileId=" + fileId,
  5451. url = config.$tcloud.api.getLink + '?fileId=' + fileId;
  5452. if (item.shareId) {
  5453. o = "AccessToken=" + token + "&Timestamp=" + time + "&dt=1&fileId=" + fileId + "&shareId=" + item.shareId;
  5454. url += '&dt=1&shareId=' + item.shareId;
  5455. }
  5456. let sign = md5(o).toString();
  5457. let res = await base.get(url, { "Accept": "application/json;charset=UTF-8", "Sign-Type": 1, "Accesstoken": token, "Timestamp": time, "Signature": sign });
  5458. if (res.res_code === 0) {
  5459. return {
  5460. index,
  5461. downloadUrl: res.fileDownloadUrl
  5462. };
  5463. } else if (res.errorCode === 'InvalidSessionKey') {
  5464. return {
  5465. index,
  5466. downloadUrl: '提示:<br/>请先登录网盘~'
  5467. };
  5468. } else if (res.res_code === 'ShareNotFoundFlatDir') {
  5469. return {
  5470. index,
  5471. downloadUrl: '提示:<br/>请[转存]文件,之后再👉前往[我的网盘]中下载哦~'
  5472. };
  5473. } else {
  5474. return {
  5475. index,
  5476. downloadUrl: '获取下载地址失败,刷新后再试试吧~' + (res.res_code ? res.res_code : "")
  5477. };
  5478. }
  5479. } catch (e) {
  5480. return {
  5481. index,
  5482. downloadUrl: '获取下载地址失败,刷新后再试试吧~'
  5483. };
  5484. }
  5485. },
  5486. async getLink() {
  5487. Swal.fire({
  5488. ...temp.swalDefault,
  5489. showConfirmButton: false,
  5490. allowOutsideClick: false,
  5491. allowEscapeKey: false,
  5492. allowEnterKey: false,
  5493. title: "获取中",
  5494. html: `...`,
  5495. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  5496. customClass: {
  5497. popup: 'loading-popup',
  5498. header: 'loading-header',
  5499. title: 'loading-title',
  5500. content: 'loading-content',
  5501. input: 'loading-input',
  5502. footer: 'loading-footer'
  5503. },
  5504. willOpen: function () {
  5505. Swal.showLoading();
  5506. },
  5507. });
  5508. let selectList = this.getSelectedList();
  5509. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  5510. if (selectList.every(item => item.isFolder)) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  5511. selectList = selectList.filter(item => !item.isFolder)
  5512. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  5513. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取状态~</div>`);
  5514. let token = base.getStorage('accessToken') || await this.getToken();
  5515. if (!token) {
  5516. return message.error('提示:<br/>请先登录网盘~');
  5517. }
  5518. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  5519. $doc.find('.loading-popup .swal2-html-container').html(`<div>获取缓存成功~</div>`);
  5520. let batchSize = 15;
  5521. let processed = 0;
  5522. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  5523. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取文件对应的下载链接~</div>`);
  5524. for (let i = 0; i < selectList.length; i += batchSize) {
  5525. let batch = selectList.slice(i, i + batchSize);
  5526. let queue = [];
  5527. batch.forEach((item, localIndex) => {
  5528. let globalIndex = i + localIndex;
  5529. queue.push(this.getFileUrlByOnce(item, globalIndex, token)
  5530. .then(val => {
  5531. processed++;
  5532. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  5533. return val;
  5534. }));
  5535. });
  5536. let res = await Promise.all(queue);
  5537. res.forEach(val => {
  5538. selectList[val.index].downloadUrl = val.downloadUrl;
  5539. });
  5540. await base.sleep(1000);
  5541. }
  5542. temp.links = [selectList, {
  5543. isFolder: v => v.isFolder,
  5544. getFileName: v => v.fileName,
  5545. getFileSize: v => v.size,
  5546. getFileLink: v => v.downloadUrl,
  5547. tooltip: config.$mcloud.dom
  5548. }];
  5549. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  5550. },
  5551. getSelectedList() {
  5552. try {
  5553. return document.querySelector(".c-file-list").__vue__.selectedList;
  5554. } catch (e) {
  5555. return [document.querySelector(".info-detail").__vue__.fileDetail];
  5556. }
  5557. },
  5558. detectPage() {
  5559. let path = location.pathname;
  5560. if (/^\/web\/main/.test(path)) return 'home';
  5561. if (/^\/web\/share/.test(path)) return 'share';
  5562. return "";
  5563. },
  5564. async initPanLinker() {
  5565. base.createTip();
  5566. base.registerMenuCommand();
  5567. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  5568. this.addButton();
  5569. } else {
  5570. this.addInitButton();
  5571. }
  5572. this.addPageListener();
  5573. this.getToken();
  5574. },
  5575. };
  5576. /**
  5577. * 迅雷云盘
  5578. * @author 油小猴
  5579. * @author hmjz100
  5580. */
  5581. let $xunlei = {
  5582. addPageListener() {
  5583. $doc.on('click', '.pl-button-mode', async function (e) {
  5584. temp.mode = e.currentTarget.dataset.mode;
  5585. if (!temp.mode) return;
  5586. $xunlei.getLink();
  5587. });
  5588. $doc.on('click', '.pl-button-save', async function (e) {
  5589. e.preventDefault();
  5590. let selectList = $xunlei.getSelectedList();
  5591. if (selectList.length === 0) {
  5592. return message.error('提示:<br/>请勾选要保存到网盘的文件哦~');
  5593. }
  5594. message.info('提示:<br/>因网盘限制,请保存到自己网盘后再去下载哦~');
  5595. await base.sleep(500);
  5596. document.querySelector('.saveToCloud').click();
  5597. });
  5598. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  5599. e.preventDefault();
  5600. let o = base._EventFactory(e);
  5601. let $width = o.item.find('.pl-progress-inner');
  5602. let $text = o.item.find('.pl-progress-inner-text');
  5603. let filename = o.link[0].dataset.filename;
  5604. let index = o.link[0].dataset.index;
  5605. let size = Number(o.link[0].dataset.size) || 0;
  5606. let originalHtml = o.link.html();
  5607. base._resetData(index);
  5608. base.get(e.currentTarget.dataset.link, undefined, 'blob', { filename, index });
  5609. let startTime = Date.now();
  5610. let prevLoaded = 0;
  5611. let prevTime = startTime;
  5612. temp.ins[index] = setInterval(async function () {
  5613. let prog = +temp.progress[index] || 0;
  5614. let currentTime = Date.now();
  5615. let elapsedTime = currentTime - startTime;
  5616. let loaded = prog * size / 100;
  5617. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  5618. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  5619. // 计算剩余时间(保护除零)
  5620. let totalProgress = Math.max(prog / 100, 0.01);
  5621. let totalElapsedSeconds = elapsedTime / 1000;
  5622. let estTotalTime = totalElapsedSeconds / totalProgress;
  5623. let remainingTime = estTotalTime - totalElapsedSeconds;
  5624. // 更新界面状态
  5625. o.link.hide();
  5626. o.directLink.hide();
  5627. o.tip.hide();
  5628. o.stop.show();
  5629. o.copy.hide();
  5630. o.progress.show();
  5631. // 更新进度条
  5632. $width.css('width', `${prog}%`);
  5633. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  5634. // 更新历史值
  5635. prevLoaded = loaded;
  5636. prevTime = currentTime;
  5637. // 下载完成
  5638. if (prog >= 100) {
  5639. await base.sleep(1000);
  5640. clearInterval(temp.ins[index]);
  5641. temp.progress[index] = 0;
  5642. o.item.find('.pl-progress-stop').hide();
  5643. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  5644. o.back.show();
  5645. await base.sleep(3000);
  5646. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  5647. }
  5648. }, 500);
  5649. });
  5650. $doc.on('click', '.listener-aria2-download', async function (e) {
  5651. let target = $(e.currentTarget);
  5652. if (target.attr('data-processing') === 'true') return;
  5653. target.attr('data-processing', 'true');
  5654. let originalHtml = target.html();
  5655. target.find(".pl-icon").remove();
  5656. target.find('.pl-loading').remove();
  5657. target.prepend(base.createLoading());
  5658. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"));
  5659. if (res === 'success') {
  5660. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  5661. } else {
  5662. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  5663. }
  5664. await base.sleep(3000);
  5665. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  5666. });
  5667. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  5668. let target = $(e.currentTarget);
  5669. if (target.attr('data-processing') === 'true') return;
  5670. target.attr('data-processing', 'true');
  5671. let originalHtml = target.html();
  5672. target.find(".pl-icon").remove();
  5673. target.find('.pl-loading').remove();
  5674. target.prepend(base.createLoading());
  5675. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"), { "mirror_url_list": base.getMirrorList(target.data("link"), config.$xunlei.api.mirror), "checkboxCustomHeadersForMirrors": "on" });
  5676. if (res === 'success') {
  5677. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  5678. } else {
  5679. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  5680. }
  5681. await base.sleep(3000);
  5682. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  5683. });
  5684. $doc.on('click', '.listener-abdm-download', async function (e) {
  5685. let target = $(e.currentTarget);
  5686. if (target.attr('data-processing') === 'true') return;
  5687. target.attr('data-processing', 'true');
  5688. let originalHtml = target.html();
  5689. target.find(".pl-icon").remove();
  5690. target.find('.pl-loading').remove();
  5691. target.prepend(base.createLoading());
  5692. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), undefined);
  5693. if (res === 'success') {
  5694. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  5695. } else {
  5696. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  5697. }
  5698. await base.sleep(3000);
  5699. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  5700. });
  5701. },
  5702. beautifyPage() {
  5703. base.adaptiveThemeOverride([
  5704. ['#3f85ff', temp.color],
  5705. ['63,133,255,.1', base.hexToRgba(`${temp.color}20`)],
  5706. ['#2670ea', `${temp.color}D0`],
  5707. ['#619bff', `${temp.color}D0`],
  5708. ['#ecf3ff', `${temp.color}10`],
  5709. ['#f6faff', `${temp.color}10`],
  5710. ['#1a2845', `${temp.color}20`],
  5711. ['#0f2035', `${temp.color}20`],
  5712. ['#308bfd', `${temp.color}20`],
  5713. ['#eee', `${temp.color}20`],
  5714. ], "other");
  5715. base.addStyle(`${mount}-xunlei`, 'style', `.web-header{background:linear-gradient(0deg,${temp.color}D0,${temp.color})}`);
  5716. },
  5717. addButton() {
  5718. base.waitForKeyElements(config.$xunlei.mount.home, (element) => {
  5719. temp.pege = $xunlei.detectPage();
  5720. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  5721. let $button = $(`<div class="xunlei-button pl-button"><i class="xlpfont xlp-download"></i><span style="font-size:13px;margin-left:6px;">下载助手</span>
  5722. <ul class="pl-dropdown-menu" style="top:34px;">
  5723. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  5724. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  5725. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  5726. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  5727. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  5728. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  5729. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  5730. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  5731. </ul>
  5732. </div>`);
  5733. element.prepend($button);
  5734. })
  5735. base.waitForKeyElements(config.$xunlei.mount.share, (element) => {
  5736. temp.pege = $xunlei.detectPage();
  5737. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  5738. let $button = $(`<div class="xunlei-button pl-button">
  5739. <i class="xlpfont xlp-download"></i><span style="font-size:13px;margin-left:6px;">下载助手</span>
  5740. <ul class="pl-dropdown-menu" style="top:34px;">
  5741. <li class="pl-button-mode pl-button-save"><i class="xlpfont xlp-file-upload"></i><span style="margin-left:3px;">转存后下载</span></li>
  5742. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  5743. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  5744. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  5745. </ul>
  5746. </div>`);
  5747. $button.css({ 'margin-right': '10px' });
  5748. element.prepend($button);
  5749. })
  5750. },
  5751. addInitButton() {
  5752. 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>`);
  5753. $button.click(base.showInitDialog);
  5754. base.waitForKeyElements(config.$xunlei.mount.home, (element) => {
  5755. temp.pege = $xunlei.detectPage();
  5756. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  5757. element.prepend($button);
  5758. })
  5759. base.waitForKeyElements(config.$xunlei.mount.share, (element) => {
  5760. temp.pege = $xunlei.detectPage();
  5761. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  5762. $button.css({ 'margin-right': '10px' });
  5763. element.prepend($button);
  5764. })
  5765. },
  5766. getToken() {
  5767. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  5768. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取状态~</div>`);
  5769. let credentials = {}, captcha = {};
  5770. for (let i = 0; i < localStorage.length; i++) {
  5771. if (/^credentials_/.test(localStorage.key(i))) {
  5772. credentials = base.getStorage(localStorage.key(i));
  5773. base.setStorage("");
  5774. }
  5775. if (/^captcha_[\w]{16}/.test(localStorage.key(i))) {
  5776. captcha = base.getStorage(localStorage.key(i));
  5777. }
  5778. }
  5779. let deviceid = /(\w{32})/.exec(base.getStorage('deviceid').split(','))[0];
  5780. let token = {
  5781. credentials,
  5782. captcha,
  5783. deviceid
  5784. };
  5785. return token;
  5786. },
  5787. async getFileUrlByOnce(item, index, token) {
  5788. try {
  5789. if (item.downloadUrl) return {
  5790. index,
  5791. downloadUrl: item.downloadUrl
  5792. };
  5793. let res = await base.get(config.$xunlei.api.getLink + item.id, { "Authorization": `${token.credentials.token_type} ${token.credentials.access_token}`, "Content-Type": "application/json", "X-Captcha-Token": token.captcha.token, "X-Device-Id": token.deviceid });
  5794. if (res.web_content_link) {
  5795. return {
  5796. index,
  5797. downloadUrl: res.web_content_link
  5798. };
  5799. } else {
  5800. return {
  5801. index,
  5802. downloadUrl: '获取下载地址失败,刷新后再试试吧~'
  5803. };
  5804. }
  5805. } catch (e) {
  5806. return message.error('提示:<br/>请先登录网盘后再刷新页面呢~');
  5807. }
  5808. },
  5809. async getLink() {
  5810. Swal.fire({
  5811. ...temp.swalDefault,
  5812. showConfirmButton: false,
  5813. allowOutsideClick: false,
  5814. allowEscapeKey: false,
  5815. allowEnterKey: false,
  5816. title: "获取中",
  5817. html: `...`,
  5818. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  5819. customClass: {
  5820. popup: 'loading-popup',
  5821. header: 'loading-header',
  5822. title: 'loading-title',
  5823. content: 'loading-content',
  5824. input: 'loading-input',
  5825. footer: 'loading-footer'
  5826. },
  5827. willOpen: function () {
  5828. Swal.showLoading();
  5829. },
  5830. });
  5831. let selectList = this.getSelectedList();
  5832. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  5833. if (selectList.every(item => item.kind !== 'drive#file')) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  5834. if (temp.pege === 'home') {
  5835. let token = this.getToken();
  5836. let batchSize = 15;
  5837. let processed = 0;
  5838. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  5839. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取文件对应的下载链接~</div>`);
  5840. for (let i = 0; i < selectList.length; i += batchSize) {
  5841. let batch = selectList.slice(i, i + batchSize);
  5842. let queue = [];
  5843. batch.forEach((item, localIndex) => {
  5844. let globalIndex = i + localIndex;
  5845. queue.push(this.getFileUrlByOnce(item, globalIndex, token)
  5846. .then(val => {
  5847. processed++;
  5848. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  5849. return val;
  5850. }));
  5851. });
  5852. let res = await Promise.all(queue);
  5853. res.forEach(val => {
  5854. selectList[val.index].downloadUrl = val.downloadUrl;
  5855. });
  5856. await base.sleep(1000);
  5857. }
  5858. } else {
  5859. return message.error('提示:<br/>页面错误~');
  5860. }
  5861. temp.links = [selectList, {
  5862. isFolder: v => v.kind === 'drive#folder',
  5863. getFileName: v => v.name,
  5864. getFileSize: v => v.size,
  5865. getFileLink: v => v.downloadUrl,
  5866. getFileMirror: v => base.getMirrorList(v, config.$xunlei.api.mirror),
  5867. tooltip: config.$xunlei.dom
  5868. }];
  5869. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  5870. },
  5871. getSelectedList() {
  5872. try {
  5873. let doms = document.querySelectorAll('.SourceListItem__item--XxpOC');
  5874. let selectedList = [];
  5875. for (let dom of doms) {
  5876. let domVue = dom.__vue__;
  5877. if (domVue.selected.includes(domVue.info.id)) {
  5878. selectedList.push(domVue.info);
  5879. }
  5880. }
  5881. return selectedList;
  5882. } catch (e) {
  5883. return [];
  5884. }
  5885. },
  5886. detectPage() {
  5887. let path = location.pathname;
  5888. if (/^\/$/.test(path)) return 'home';
  5889. if (/^\/(s|share)\//.test(path)) return 'share';
  5890. return "";
  5891. },
  5892. async initPanLinker() {
  5893. base.createTip();
  5894. base.registerMenuCommand();
  5895. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  5896. this.addButton();
  5897. } else {
  5898. this.addInitButton();
  5899. }
  5900. this.addPageListener();
  5901. },
  5902. };
  5903. /**
  5904. * 夸克网盘
  5905. * @author 油小猴
  5906. * @author hmjz100
  5907. */
  5908. let $quark = {
  5909. addPageListener() {
  5910. $doc.on('click', '.pl-button-mode', async function (e) {
  5911. temp.mode = e.currentTarget.dataset.mode;
  5912. if (!temp.mode) return;
  5913. $quark.getLink();
  5914. });
  5915. $doc.on('click', '.pl-button-save', async function (e) {
  5916. e.preventDefault();
  5917. let selectList = $quark.getSelectedList();
  5918. if (selectList.length === 0) {
  5919. return message.error('提示:<br/>请勾选要保存到网盘的文件哦~');
  5920. }
  5921. message.info('提示:<br/>因网盘限制,请保存到自己网盘后再去下载哦~');
  5922. await base.sleep(500);
  5923. document.querySelector('.share-path').click();
  5924. base.waitForKeyElements(".btn-file.btn-file-primary.confirm-btn", (element) => {
  5925. element.one("click", async () => {
  5926. await base.sleep(1000);
  5927. document.querySelector('.share-save').click();
  5928. })
  5929. return true;
  5930. }, true)
  5931. });
  5932. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  5933. e.preventDefault();
  5934. let o = base._EventFactory(e);
  5935. let $width = o.item.find('.pl-progress-inner');
  5936. let $text = o.item.find('.pl-progress-inner-text');
  5937. let filename = o.link[0].dataset.filename;
  5938. let index = o.link[0].dataset.index;
  5939. let size = Number(o.link[0].dataset.size) || 0;
  5940. let originalHtml = o.link.html();
  5941. base._resetData(index);
  5942. base.get(e.currentTarget.dataset.link, { "User-Agent": config.$quark.api.ua.downloadLink }, 'blob', { filename, index });
  5943. let startTime = Date.now();
  5944. let prevLoaded = 0;
  5945. let prevTime = startTime;
  5946. temp.ins[index] = setInterval(async function () {
  5947. let prog = +temp.progress[index] || 0;
  5948. let currentTime = Date.now();
  5949. let elapsedTime = currentTime - startTime;
  5950. let loaded = prog * size / 100;
  5951. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  5952. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  5953. // 计算剩余时间(保护除零)
  5954. let totalProgress = Math.max(prog / 100, 0.01);
  5955. let totalElapsedSeconds = elapsedTime / 1000;
  5956. let estTotalTime = totalElapsedSeconds / totalProgress;
  5957. let remainingTime = estTotalTime - totalElapsedSeconds;
  5958. // 更新界面状态
  5959. o.link.hide();
  5960. o.directLink.hide();
  5961. o.tip.hide();
  5962. o.stop.show();
  5963. o.copy.hide();
  5964. o.progress.show();
  5965. // 更新进度条
  5966. $width.css('width', `${prog}%`);
  5967. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  5968. // 更新历史值
  5969. prevLoaded = loaded;
  5970. prevTime = currentTime;
  5971. // 下载完成
  5972. if (prog >= 100) {
  5973. await base.sleep(1000);
  5974. clearInterval(temp.ins[index]);
  5975. temp.progress[index] = 0;
  5976. o.item.find('.pl-progress-stop').hide();
  5977. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  5978. o.back.show();
  5979. await base.sleep(3000);
  5980. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  5981. }
  5982. }, 500);
  5983. });
  5984. $doc.on('click', '.listener-aria2-download', async function (e) {
  5985. let target = $(e.currentTarget);
  5986. if (target.attr('data-processing') === 'true') return;
  5987. target.attr('data-processing', 'true');
  5988. let originalHtml = target.html();
  5989. target.find(".pl-icon").remove();
  5990. target.find('.pl-loading').remove();
  5991. target.prepend(base.createLoading());
  5992. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"), [`User-Agent:${config.$quark.api.ua.downloadLink}`, `Referer:https://${location.host}/`, `Cookie:${document.cookie}`]);
  5993. if (res === 'success') {
  5994. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  5995. } else {
  5996. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  5997. }
  5998. await base.sleep(3000);
  5999. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6000. });
  6001. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  6002. let target = $(e.currentTarget);
  6003. if (target.attr('data-processing') === 'true') return;
  6004. target.attr('data-processing', 'true');
  6005. let originalHtml = target.html();
  6006. target.find(".pl-icon").remove();
  6007. target.find('.pl-loading').remove();
  6008. target.prepend(base.createLoading());
  6009. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"), { "user_agent": config.$quark.api.ua.downloadLink, "referrer": `https://${location.host}/`, "cookie": document.cookie });
  6010. if (res === 'success') {
  6011. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6012. } else {
  6013. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6014. }
  6015. await base.sleep(3000);
  6016. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6017. });
  6018. $doc.on('click', '.listener-abdm-download', async function (e) {
  6019. let target = $(e.currentTarget);
  6020. if (target.attr('data-processing') === 'true') return;
  6021. target.attr('data-processing', 'true');
  6022. let originalHtml = target.html();
  6023. target.find(".pl-icon").remove();
  6024. target.find('.pl-loading').remove();
  6025. target.prepend(base.createLoading());
  6026. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), { "User-Agent": config.$quark.api.ua.downloadLink, "Cookie": document.cookie });
  6027. if (res === 'success') {
  6028. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6029. } else {
  6030. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6031. }
  6032. await base.sleep(3000);
  6033. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6034. });
  6035. },
  6036. greenerPage() {
  6037. base.waitForKeyElements('[class*="Activity--video-toolbar-activity"]', function (tag) {
  6038. tag.fadeOut();
  6039. }, true);
  6040. base.waitForKeyElements('span[class*="SectionHeaderController--icon-download"]', function (tag) {
  6041. tag.fadeOut();
  6042. }, true);
  6043. base.waitForKeyElements('div[class*="SectionHeaderController--download-popover"]', function (tag) {
  6044. tag.find(".ant-popover-arrow").css({ "left": "75%" });
  6045. }, true);
  6046. base.waitForKeyElements('div[class*="DetailLayout--client-download"]', function (tag) {
  6047. tag.fadeOut();
  6048. }, true);
  6049. base.waitForKeyElements(".next-box.share-right-side-content", function (tag) {
  6050. tag.fadeOut();
  6051. }, true);
  6052. base.waitForKeyElements('[class*="DetailLayout--container"] .feature-screen', function (tag) {
  6053. tag.fadeOut();
  6054. }, true);
  6055. base.waitForKeyElements('.ant-modal-content .ant-modal-body .right-wrap', function (tag) {
  6056. if (tag.find(".hint").text().includes("客户端")) tag.fadeOut();
  6057. }, true);
  6058. base.waitForKeyElements(".pc-member-entrance span.button-text", function (tag) {
  6059. tag.text("会员中心");
  6060. let observer = new MutationObserver(function (mutations) {
  6061. mutations.forEach(function (mutation) {
  6062. if (tag.text() === "会员中心") return
  6063. tag.text("会员中心");
  6064. });
  6065. });
  6066. let config = { subtree: true, characterData: true, childList: true };
  6067. observer.observe(tag[0], config);
  6068. }, true);
  6069. base.waitForKeyElements(".pc-member-entrance .tips", function (tag) {
  6070. tag.fadeOut();
  6071. }, true);
  6072. base.waitForKeyElements(".modal .modal-content .halo-animated-background .halo-content .pay-modal .close", function (tag) {
  6073. tag[0].click();
  6074. }, true);
  6075. base.waitForKeyElements(".modal .modal-content .halo-animated-background .halo-content .red-envelope .close", function (tag) {
  6076. tag[0].click();
  6077. }, true);
  6078. },
  6079. beautifyPage() {
  6080. base.adaptiveThemeOverride([
  6081. ['#0d53ff', temp.color],
  6082. ['#e6f1ff', `${temp.color}20`],
  6083. ['#f0faff', `${temp.color}20`],
  6084. ['#7da3ff', `${temp.color}D0`],
  6085. ['#ddd', `${temp.color}D0`],
  6086. ['17,17,17,.9', base.hexToRgba(`${temp.color}D0`)],
  6087. ['40,40,255,.04', base.hexToRgba(`${temp.color}20`)],
  6088. ['#f7f7ff', 'transparent'],
  6089. ['238,247,255,0', base.hexToRgba(`${temp.color}00`)],
  6090. ]);
  6091. base.addStyle(`${mount}-quark`, 'style', `.file-list .hover-oper .hover-transparent-bg{background:transparent!important} .ant-checkbox-wrapper .ant-checkbox-checked .ant-checkbox-inner,.ant-checkbox-wrapper .ant-checkbox-indeterminate .ant-checkbox-inner:after{background-color:${temp.color}!important}`);
  6092. },
  6093. svg: '',
  6094. addButton() {
  6095. base.waitForKeyElements(config.$quark.mount.home, (element) => {
  6096. temp.pege = $quark.detectPage();
  6097. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  6098. let $button = $(`<div class="ant-dropdown-trigger pl-button" style="display: inline-block;">
  6099. <div class="ant-upload ant-upload-select ant-upload-select-text">
  6100. <ul class="pl-dropdown-menu" style="top:35px">
  6101. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  6102. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  6103. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  6104. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  6105. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  6106. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  6107. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  6108. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  6109. </ul>
  6110. <button type="button" class="ant-btn ant-btn-primary quark-button">
  6111. <img class="btn-icon" src="${$quark.svg}">
  6112. <span>下载助手</span>
  6113. </button>
  6114. </div>
  6115. </div>`);
  6116. $button.css({ "margin-right": "16px" });
  6117. element.prepend($button);
  6118. })
  6119. base.waitForKeyElements(config.$quark.mount.share, (element) => {
  6120. temp.pege = $quark.detectPage();
  6121. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  6122. let $button = $(`<button type="button" class="ant-btn btn-file ant-btn-primary pl-button quark-button"><img class="btn-icon" src="${$quark.svg}"><span>下载助手</span><ul class="pl-dropdown-menu" style="bottom:22px;left:0"><li class="pl-button-mode pl-button-save"><span class="share-save-ico"></span>保存后下载</li><li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li><li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li><li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li></ul></button>`);
  6123. $button.css({ "height": "36px", "margin-left": "16px", "border-radius": "6px", "display": "inline-block" });
  6124. element.append($button);
  6125. })
  6126. },
  6127. addInitButton() {
  6128. base.waitForKeyElements(config.$quark.mount.home, (element) => {
  6129. temp.pege = $quark.detectPage();
  6130. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  6131. let $button = $(`<button type="button" class="ant-btn ant-btn-primary quark-button pl-button-init"><img class="btn-icon" src="${$quark.svg}"><span>点我点亮</span></button>`);
  6132. $button.css({ "margin-right": "16px", "display": "inline-block" });
  6133. $button.click(base.showInitDialog);
  6134. element.prepend($button);
  6135. })
  6136. base.waitForKeyElements(config.$quark.mount.share, (element) => {
  6137. temp.pege = $quark.detectPage();
  6138. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  6139. let $button = $(`<button type="button" class="ant-btn btn-file ant-btn-primary pl-button-init quark-button"><img class="btn-icon" src="${$quark.svg}"><span>点我点亮</span></button>`);
  6140. $button.css({ "height": "36px", "margin-left": "16px", "border-radius": "6px", "display": "inline-block" });
  6141. $button.click(base.showInitDialog);
  6142. element.append($button);
  6143. })
  6144. },
  6145. async getLink() {
  6146. Swal.fire({
  6147. ...temp.swalDefault,
  6148. showConfirmButton: false,
  6149. allowOutsideClick: false,
  6150. allowEscapeKey: false,
  6151. allowEnterKey: false,
  6152. title: "获取中",
  6153. html: `...`,
  6154. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  6155. customClass: {
  6156. popup: 'loading-popup',
  6157. header: 'loading-header',
  6158. title: 'loading-title',
  6159. content: 'loading-content',
  6160. input: 'loading-input',
  6161. footer: 'loading-footer'
  6162. },
  6163. willOpen: function () {
  6164. Swal.showLoading();
  6165. },
  6166. });
  6167. let selectList = this.getSelectedList();
  6168. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  6169. if (selectList.every(item => !item.file)) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  6170. if (temp.pege === 'home') {
  6171. let data = [];
  6172. let batchSize = 15;
  6173. let processed = 0;
  6174. selectList = selectList.filter(item => item.file === true)
  6175. for (let i = 0; i < selectList.length; i += batchSize) {
  6176. // 获取当前批次文件
  6177. let batch = selectList.slice(i, i + batchSize);
  6178. let fids = batch.map(item => item.fid);
  6179. // 发起请求获取链接
  6180. let res = await base.post(config.$quark.api.getLink, { "fids": fids }, { "User-Agent": config.$quark.api.ua.downloadLink });
  6181. if (res?.code === 31001) {
  6182. return message.error('提示:<br/>请先登录网盘~<br/>代码:' + res.code);
  6183. }
  6184. if (res?.code !== 0) {
  6185. return message.error('提示:<br/>获取链接失败了~<br/>代码:' + res.code);
  6186. }
  6187. // 合并响应数据
  6188. if (res?.data) {
  6189. data.push(...res.data);
  6190. }
  6191. // 更新处理进度
  6192. processed += batch.length;
  6193. // 更新UI显示
  6194. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  6195. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  6196. // 请求间隔节流
  6197. await base.sleep(1000);
  6198. }
  6199. temp.links = [data, {
  6200. isFolder: v => v.file === false,
  6201. getFileName: v => v.file_name,
  6202. getFileSize: v => v.size,
  6203. getFileLink: v => v.download_url,
  6204. convert: {
  6205. aria2: `--header "User-Agent:${config.$quark.api.ua.downloadLink}" --header "Referer:https://${location.host}/" --header "Cookie:${document.cookie}"`,
  6206. curl: `-A "${config.$quark.api.ua.downloadLink}" -e "https://${location.host}/" -b "${document.cookie}"`,
  6207. bitcomet: `user_agent=${encodeURIComponent(config.$quark.api.ua.downloadLink)}&refer=${encodeURIComponent(`https://${location.host}/`)}&cookie=${encodeURIComponent(document.cookie)}`
  6208. },
  6209. tooltip: config.$quark.dom
  6210. }];
  6211. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  6212. } else {
  6213. return message.error('提示:<br/>页面错误~');
  6214. }
  6215. },
  6216. getSelectedList() {
  6217. try {
  6218. let selectedList = [];
  6219. let reactDom = document.getElementsByClassName('file-list')[0];
  6220. let reactObj = base.findReact(reactDom);
  6221. let props = reactObj.props;
  6222. if (props) {
  6223. let fileList = props.list || [];
  6224. let selectedKeys = props.selectedRowKeys || [];
  6225. fileList.forEach(function (val) {
  6226. if (selectedKeys.includes(val.fid)) {
  6227. selectedList.push(val);
  6228. }
  6229. });
  6230. }
  6231. return selectedList;
  6232. } catch (e) {
  6233. return [];
  6234. }
  6235. },
  6236. detectPage() {
  6237. let path = location.pathname;
  6238. if (/^\/(list)/.test(path)) return 'home';
  6239. if (/^\/(s|share)\//.test(path)) return 'share';
  6240. return "";
  6241. },
  6242. async initPanLinker() {
  6243. base.createTip();
  6244. base.registerMenuCommand();
  6245. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  6246. this.addButton();
  6247. } else {
  6248. this.addInitButton();
  6249. }
  6250. this.addPageListener();
  6251. },
  6252. };
  6253. /**
  6254. * UC网盘
  6255. * @author 油小猴
  6256. * @author hmjz100
  6257. */
  6258. let $uc = {
  6259. addPageListener() {
  6260. $doc.on('click', '.pl-button-mode', async function (e) {
  6261. temp.mode = e.currentTarget.dataset.mode;
  6262. if (!temp.mode) return;
  6263. $uc.getLink();
  6264. });
  6265. $doc.on('click', '.pl-button-save', async function (e) {
  6266. e.preventDefault();
  6267. let selectList = $uc.getSelectedList();
  6268. if (selectList.length === 0) {
  6269. return message.error('提示:<br/>请勾选要保存到网盘的文件哦~');
  6270. }
  6271. message.info('提示:<br/>因网盘限制,请保存到自己网盘后再去下载哦~');
  6272. await base.sleep(500);
  6273. document.querySelector('.file-info_r').click();
  6274. });
  6275. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  6276. e.preventDefault();
  6277. let o = base._EventFactory(e);
  6278. let $width = o.item.find('.pl-progress-inner');
  6279. let $text = o.item.find('.pl-progress-inner-text');
  6280. let filename = o.link[0].dataset.filename;
  6281. let index = o.link[0].dataset.index;
  6282. let size = Number(o.link[0].dataset.size) || 0;
  6283. let originalHtml = o.link.html();
  6284. base._resetData(index);
  6285. base.get(e.currentTarget.dataset.link, { "User-Agent": config.$uc.api.ua.downloadLink }, 'blob', { filename, index });
  6286. let startTime = Date.now();
  6287. let prevLoaded = 0;
  6288. let prevTime = startTime;
  6289. temp.ins[index] = setInterval(async function () {
  6290. let prog = +temp.progress[index] || 0;
  6291. let currentTime = Date.now();
  6292. let elapsedTime = currentTime - startTime;
  6293. let loaded = prog * size / 100;
  6294. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  6295. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  6296. // 计算剩余时间(保护除零)
  6297. let totalProgress = Math.max(prog / 100, 0.01);
  6298. let totalElapsedSeconds = elapsedTime / 1000;
  6299. let estTotalTime = totalElapsedSeconds / totalProgress;
  6300. let remainingTime = estTotalTime - totalElapsedSeconds;
  6301. // 更新界面状态
  6302. o.link.hide();
  6303. o.directLink.hide();
  6304. o.tip.hide();
  6305. o.stop.show();
  6306. o.copy.hide();
  6307. o.progress.show();
  6308. // 更新进度条
  6309. $width.css('width', `${prog}%`);
  6310. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  6311. // 更新历史值
  6312. prevLoaded = loaded;
  6313. prevTime = currentTime;
  6314. // 下载完成
  6315. if (prog >= 100) {
  6316. await base.sleep(1000);
  6317. clearInterval(temp.ins[index]);
  6318. temp.progress[index] = 0;
  6319. o.item.find('.pl-progress-stop').hide();
  6320. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  6321. o.back.show();
  6322. await base.sleep(3000);
  6323. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  6324. }
  6325. }, 500);
  6326. });
  6327. $doc.on('click', '.listener-aria2-download', async function (e) {
  6328. let target = $(e.currentTarget);
  6329. if (target.attr('data-processing') === 'true') return;
  6330. target.attr('data-processing', 'true');
  6331. let originalHtml = target.html();
  6332. target.find(".pl-icon").remove();
  6333. target.find('.pl-loading').remove();
  6334. target.prepend(base.createLoading());
  6335. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"), [`User-Agent:${config.$uc.api.ua.downloadLink}`, `Referer:https://${location.host}/`, `Cookie:${document.cookie}`]);
  6336. if (res === 'success') {
  6337. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6338. } else {
  6339. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6340. }
  6341. await base.sleep(3000);
  6342. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6343. });
  6344. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  6345. let target = $(e.currentTarget);
  6346. if (target.attr('data-processing') === 'true') return;
  6347. target.attr('data-processing', 'true');
  6348. let originalHtml = target.html();
  6349. target.find(".pl-icon").remove();
  6350. target.find('.pl-loading').remove();
  6351. target.prepend(base.createLoading());
  6352. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"), { "user_agent": config.$uc.api.ua.downloadLink, "referrer": `https://${location.host}/`, "cookie": document.cookie });
  6353. if (res === 'success') {
  6354. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6355. } else {
  6356. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6357. }
  6358. await base.sleep(3000);
  6359. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6360. });
  6361. $doc.on('click', '.listener-abdm-download', async function (e) {
  6362. let target = $(e.currentTarget);
  6363. if (target.attr('data-processing') === 'true') return;
  6364. target.attr('data-processing', 'true');
  6365. let originalHtml = target.html();
  6366. target.find(".pl-icon").remove();
  6367. target.find('.pl-loading').remove();
  6368. target.prepend(base.createLoading());
  6369. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), { "User-Agent": config.$uc.api.ua.downloadLink, "Cookie": document.cookie });
  6370. if (res === 'success') {
  6371. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6372. } else {
  6373. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6374. }
  6375. await base.sleep(3000);
  6376. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6377. });
  6378. },
  6379. greenerPage() {
  6380. base.waitForKeyElements('[class*="VideoDetail--content-footer"]', function (tag) {
  6381. tag.children().each(function () {
  6382. let $child = $(this);
  6383. if ($child.text().includes('手机客户端')) {
  6384. $child.hide();
  6385. }
  6386. });
  6387. }, true);
  6388. base.waitForKeyElements('[class*="PCLandingBanner--ad-block"]', function (tag) {
  6389. tag.hide();
  6390. }, true);
  6391. },
  6392. beautifyPage() {
  6393. base.adaptiveThemeOverride([
  6394. ['#12161a', temp.color],
  6395. ['#e6f1ff', `${temp.color}20`],
  6396. ['#f0faff', `${temp.color}20`],
  6397. ['#7da3ff', `${temp.color}D0`],
  6398. ['#ddd', `${temp.color}D0`],
  6399. ['17,17,17,.9', base.hexToRgba(`${temp.color}D0`)],
  6400. ['40,40,255,.04', base.hexToRgba(`${temp.color}20`)],
  6401. ['#f7f7ff', 'transparent'],
  6402. ['238,247,255,0', base.hexToRgba(`${temp.color}00`)],
  6403. ]);
  6404. base.addStyle(`${mount}-uc`, 'style', `.file-list .hover-oper .hover-transparent-bg{background:transparent!important}`);
  6405. },
  6406. svg: '',
  6407. addButton() {
  6408. base.waitForKeyElements(config.$uc.mount.home, (element) => {
  6409. temp.pege = $uc.detectPage();
  6410. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  6411. let $button = $(`<div class="ant-dropdown-trigger pl-button">
  6412. <button type="button" class="uc-button ant-btn btn-file ant-btn-primary">
  6413. <img class="uc-btn-icon" src="${$uc.svg}"><span>下载助手</span>
  6414. </button>
  6415. <ul class="pl-dropdown-menu" style="top:39px;">
  6416. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  6417. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  6418. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  6419. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  6420. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  6421. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  6422. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  6423. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  6424. </ul>
  6425. </div>`);
  6426. $button.css({ "margin-right": "10px", "display": "inline-block" });
  6427. element.prepend($button);
  6428. })
  6429. base.waitForKeyElements(config.$uc.mount.share, (element) => {
  6430. temp.pege = $uc.detectPage();
  6431. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  6432. let $button = $(`<div class="ant-dropdown-trigger pl-button"><button type="button" class="uc-button ant-btn btn-file ant-btn-primary" style="height:40px;"><img class="uc-btn-icon" src="${$uc.svg}"><span>下载助手</span></button><ul class="pl-dropdown-menu"><li class="pl-button-mode pl-button-save"><span class="save-btn-icon"></span>保存后下载</li><li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li><li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li><li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li></ul></div>`);
  6433. $button.css({ "margin-left": "10px", "display": "inline-block" });
  6434. element.append($button);
  6435. })
  6436. },
  6437. addInitButton() {
  6438. let $button = $(`<div class="ant-dropdown-trigger pl-button-init"><button type="button" class="uc-button ant-btn btn-file ant-btn-primary" style="height:40px;"><img class="uc-btn-icon" src="${$uc.svg}"><span>点我点亮</span></button></div>`);
  6439. $button.click(base.showInitDialog);
  6440. base.waitForKeyElements(config.$uc.mount.home, (element) => {
  6441. temp.pege = $uc.detectPage();
  6442. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  6443. $button.css({ "margin-right": "10px", "display": "inline-block" });
  6444. element.prepend($button);
  6445. })
  6446. base.waitForKeyElements(config.$uc.mount.share, (element) => {
  6447. temp.pege = $uc.detectPage();
  6448. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  6449. $button.css({ "margin-left": "10px", "display": "inline-block" });
  6450. element.append($button);
  6451. })
  6452. },
  6453. async getLink() {
  6454. Swal.fire({
  6455. ...temp.swalDefault,
  6456. showConfirmButton: false,
  6457. allowOutsideClick: false,
  6458. allowEscapeKey: false,
  6459. allowEnterKey: false,
  6460. title: "获取中",
  6461. html: `...`,
  6462. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  6463. customClass: {
  6464. popup: 'loading-popup',
  6465. header: 'loading-header',
  6466. title: 'loading-title',
  6467. content: 'loading-content',
  6468. input: 'loading-input',
  6469. footer: 'loading-footer'
  6470. },
  6471. willOpen: function () {
  6472. Swal.showLoading();
  6473. },
  6474. });
  6475. let selectList = this.getSelectedList();
  6476. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  6477. if (selectList.every(item => !item.file)) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  6478. if (temp.pege === 'home') {
  6479. let data = [];
  6480. let batchSize = 15;
  6481. let processed = 0;
  6482. selectList = selectList.filter(item => item.file === true)
  6483. for (let i = 0; i < selectList.length; i += batchSize) {
  6484. // 获取当前批次文件
  6485. let batch = selectList.slice(i, i + batchSize);
  6486. let fids = batch.map(item => item.fid);
  6487. // 发起请求获取链接
  6488. let res = await base.post(config.$uc.api.getLink, { "fids": fids }, { "User-Agent": config.$uc.api.ua.downloadLink });
  6489. if (res?.code === 31001) {
  6490. return message.error('提示:<br/>请先登录网盘~<br/>代码:' + res.code);
  6491. }
  6492. if (res?.code !== 0) {
  6493. return message.error('提示:<br/>获取链接失败了~<br/>代码:' + res.code);
  6494. }
  6495. // 合并响应数据
  6496. if (res?.data) {
  6497. data.push(...res.data);
  6498. }
  6499. // 更新处理进度
  6500. processed += batch.length;
  6501. // 更新UI显示
  6502. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  6503. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  6504. // 请求间隔节流
  6505. await base.sleep(1000);
  6506. }
  6507. temp.links = [data, {
  6508. isFolder: v => v.file === false,
  6509. getFileName: v => v.file_name,
  6510. getFileSize: v => v.size,
  6511. getFileLink: v => v.download_url,
  6512. convert: {
  6513. aria2: `--header "User-Agent:${config.$uc.api.ua.downloadLink}" --header "Referer:https://${location.host}/" --header "Cookie:${document.cookie}"`,
  6514. curl: `-A "${config.$uc.api.ua.downloadLink}" -e "https://${location.host}/" -b "${document.cookie}"`,
  6515. bitcomet: `user_agent=${encodeURIComponent(config.$uc.api.ua.downloadLink)}&refer=${encodeURIComponent(`https://${location.host}/`)}&cookie=${encodeURIComponent(document.cookie)}`
  6516. },
  6517. tooltip: config.$uc.dom
  6518. }];
  6519. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  6520. } else {
  6521. return message.error('提示:<br/>页面错误~');
  6522. }
  6523. },
  6524. getSelectedList() {
  6525. try {
  6526. let selectedList = [];
  6527. let reactDom = document.getElementsByClassName('file-list')[0];
  6528. let reactObj = base.findReact(reactDom);
  6529. let props = reactObj.props;
  6530. if (props) {
  6531. let fileList = props.list || [];
  6532. let selectedKeys = props.selectedRowKeys || [];
  6533. fileList.forEach(function (val) {
  6534. if (selectedKeys.includes(val.fid)) {
  6535. selectedList.push(val);
  6536. }
  6537. });
  6538. }
  6539. return selectedList;
  6540. } catch (e) {
  6541. return [];
  6542. }
  6543. },
  6544. detectPage() {
  6545. let path = location.pathname;
  6546. if (/^\/(list)/.test(path)) return 'home';
  6547. if (/^\/(s|share)\//.test(path)) return 'share';
  6548. return "";
  6549. },
  6550. async initPanLinker() {
  6551. base.createTip();
  6552. base.registerMenuCommand();
  6553. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  6554. this.addButton();
  6555. } else {
  6556. this.addInitButton();
  6557. }
  6558. this.addPageListener();
  6559. },
  6560. };
  6561. /**
  6562. * 123云盘
  6563. * @author 油小猴
  6564. * @author hmjz100
  6565. */
  6566. let $123pan = {
  6567. addPageListener() {
  6568. $doc.on('click', '.pl-button-mode', async function (e) {
  6569. temp.mode = e.currentTarget.dataset.mode;
  6570. if (!temp.mode) return;
  6571. $123pan.getLink();
  6572. });
  6573. $doc.on('click', '.listener-api-download.enhance', async function (e) {
  6574. e.preventDefault();
  6575. let o = base._EventFactory(e);
  6576. let $width = o.item.find('.pl-progress-inner');
  6577. let $text = o.item.find('.pl-progress-inner-text');
  6578. let filename = o.link[0].dataset.filename;
  6579. let index = o.link[0].dataset.index;
  6580. let size = Number(o.link[0].dataset.size) || 0;
  6581. let originalHtml = o.link.html();
  6582. base._resetData(index);
  6583. let token = $123pan.getToken();
  6584. base.get(e.currentTarget.dataset.link, undefined, 'blob', { filename, index });
  6585. let startTime = Date.now();
  6586. let prevLoaded = 0;
  6587. let prevTime = startTime;
  6588. temp.ins[index] = setInterval(async function () {
  6589. let prog = +temp.progress[index] || 0;
  6590. let currentTime = Date.now();
  6591. let elapsedTime = currentTime - startTime;
  6592. let loaded = prog * size / 100;
  6593. let timeDiff = Math.max(currentTime - prevTime, 1); // 避免除零
  6594. let speed = ((loaded - prevLoaded) / (timeDiff / 1000)) || 0;
  6595. // 计算剩余时间(保护除零)
  6596. let totalProgress = Math.max(prog / 100, 0.01);
  6597. let totalElapsedSeconds = elapsedTime / 1000;
  6598. let estTotalTime = totalElapsedSeconds / totalProgress;
  6599. let remainingTime = estTotalTime - totalElapsedSeconds;
  6600. // 更新界面状态
  6601. o.link.hide();
  6602. o.directLink.hide();
  6603. o.tip.hide();
  6604. o.stop.show();
  6605. o.copy.hide();
  6606. o.progress.show();
  6607. // 更新进度条
  6608. $width.css('width', `${prog}%`);
  6609. $text.text(`${prog.toFixed(2)}% | 速度:${base.sizeFormat(speed)} | 剩余:${base.rtimeFormat(remainingTime)}`);
  6610. // 更新历史值
  6611. prevLoaded = loaded;
  6612. prevTime = currentTime;
  6613. // 下载完成
  6614. if (prog >= 100) {
  6615. await base.sleep(1000);
  6616. clearInterval(temp.ins[index]);
  6617. temp.progress[index] = 0;
  6618. o.item.find('.pl-progress-stop').hide();
  6619. $text.text('下载完成~ 浏览器下载框应该弹出来了哦~');
  6620. o.back.show();
  6621. await base.sleep(3000);
  6622. o.link.html(originalHtml).animate({ opacity: '1' }, "slow");
  6623. }
  6624. }, 500);
  6625. });
  6626. $doc.on('click', '.listener-aria2-download', async function (e) {
  6627. let target = $(e.currentTarget);
  6628. if (target.attr('data-processing') === 'true') return;
  6629. target.attr('data-processing', 'true');
  6630. let originalHtml = target.html();
  6631. target.find(".pl-icon").remove();
  6632. target.find('.pl-loading').remove();
  6633. target.prepend(base.createLoading());
  6634. let res = await base.sendLinkToAria2(target.data("link"), target.data("filename"));
  6635. if (res === 'success') {
  6636. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6637. } else {
  6638. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6639. }
  6640. await base.sleep(3000);
  6641. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6642. });
  6643. $doc.on('click', '.listener-bitcomet-download', async function (e) {
  6644. let target = $(e.currentTarget);
  6645. if (target.attr('data-processing') === 'true') return;
  6646. target.attr('data-processing', 'true');
  6647. let originalHtml = target.html();
  6648. target.find(".pl-icon").remove();
  6649. target.find('.pl-loading').remove();
  6650. target.prepend(base.createLoading());
  6651. let res = await base.sendLinkToBitcomet(target.data("link"), target.data("filename"));
  6652. if (res === 'success') {
  6653. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6654. } else {
  6655. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6656. }
  6657. await base.sleep(3000);
  6658. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6659. });
  6660. $doc.on('click', '.listener-abdm-download', async function (e) {
  6661. let target = $(e.currentTarget);
  6662. if (target.attr('data-processing') === 'true') return;
  6663. target.attr('data-processing', 'true');
  6664. let originalHtml = target.html();
  6665. target.find(".pl-icon").remove();
  6666. target.find('.pl-loading').remove();
  6667. target.prepend(base.createLoading());
  6668. let res = await base.sendLinkToABDM(target.data("link"), target.data("filename"), undefined);
  6669. if (res === 'success') {
  6670. target.removeClass('pl-btn-danger').html('发送成功啦!快去看看吧~').animate({ opacity: '0.5' }, "slow");
  6671. } else {
  6672. target.addClass('pl-btn-danger').text('发送失败,检查一下您的配置信息哦!').animate({ opacity: '0.5' }, "slow");
  6673. }
  6674. await base.sleep(3000);
  6675. target.removeClass('pl-btn-danger').removeAttr('data-processing').html(originalHtml).css('opacity', "");
  6676. });
  6677. },
  6678. greenerPage() {
  6679. base.waitForKeyElements(".cent > .cent-not-login > .ant-btn", (tag) => {
  6680. if (tag.hasClass("reg") || tag.hasClass("log")) return;
  6681. tag.addClass("reg");
  6682. tag.removeClass("loginRight");
  6683. tag.find("span").text("注册");
  6684. if (tag.next().hasClass("log")) return;
  6685. let button = $(`<button type="button" class="ant-btn ant-btn-default ant-btn-two-chinese-chars log loginRight" style="width:auto!important;height:auto!important;margin-left:10px!important"><span>登录</span></button>`);
  6686. button.on("click", () => {
  6687. let login = new URL(`https://login.123pan.com/centerlogin`);
  6688. login.searchParams.set("redirect_url", location.href);
  6689. location.href = login;
  6690. });
  6691. tag.after(button);
  6692. });
  6693. base.waitForKeyElements(".new-menu-item-image, .special-menu-item-container-migration--label, .sider-member-btn, .video-new-user-tips", (tag) => {
  6694. if (tag.is(":hidden")) return;
  6695. tag.hide();
  6696. }, true);
  6697. base.waitForKeyElements('.frontend-layout-header-right > span > [alt^="buttonMall"]', (tag) => {
  6698. if (tag.parent().is(":hidden")) return;
  6699. tag.parent().hide();
  6700. let button = $(`<div class="frontend-layout-header-right-button-invite-new">会员中心</div>`);
  6701. button.on("click", () => { tag.click() });
  6702. tag.parent().after(button);
  6703. }, true);
  6704. let disallowTexts = ["同步空间", "其他网盘数据转入", "下载客户端"];
  6705. base.waitForKeyElements('ul[role="menu"] li[role="menuitem"]', (tag) => {
  6706. let menuText = tag.text().trim();
  6707. if (tag.is(":hidden")) return;
  6708. if (disallowTexts.includes(menuText)) tag.hide();
  6709. }, true);
  6710. base.waitForKeyElements(`.rightInfo .register:not(.pl-button, .pl-button-init),
  6711. .homeClass > div > .ant-dropdown-trigger:not(.pl-button, .pl-button-init),
  6712. .homeClass > div > .sysbut`, function (tag) {
  6713. let hasTextNode = false;
  6714. tag.contents().each(function () {
  6715. if (this.nodeType === 3 && $.trim(this.textContent)) {
  6716. hasTextNode = true;
  6717. return;
  6718. }
  6719. });
  6720. if (!hasTextNode) return;
  6721. tag.css({ "width": "38px" });
  6722. tag.contents().each(function () {
  6723. if (this.nodeType === 3) {
  6724. $(this).remove();
  6725. }
  6726. });
  6727. tag.find('svg').css({ "margin-right": "0" });
  6728. });
  6729. base.waitForKeyElements('.rightInfo .qrcode_btn', function (tag) {
  6730. tag.hide();
  6731. }, true);
  6732. },
  6733. beautifyPage() {
  6734. base.adaptiveThemeOverride([
  6735. ['#597dfc', temp.color],
  6736. ['#5a7cfc', temp.color],
  6737. ['#2A82E4', temp.color],
  6738. ['#51a1f0', temp.color],
  6739. ['#597DFC', temp.color],
  6740. ['#40a9ff', temp.color],
  6741. ['#3c80ff', temp.color],
  6742. ['#3C80FF', temp.color],
  6743. ['#1890ff', temp.color],
  6744. ['#F0F8FF', `${temp.color}10`],
  6745. ['#f0f9ff', `${temp.color}20`],
  6746. ['#F2F5FF', `${temp.color}20`],
  6747. ['#C5E1FF', `${temp.color}20`],
  6748. ['#2961D9', `${temp.color}20`],
  6749. ['#b8d8ff', `${temp.color}20`],
  6750. ['#325cf0', `${temp.color}D0`],
  6751. ['#66A1FF', `${temp.color}D0`],
  6752. ['60, 128, 255', base.hexToRgba(temp.color)],
  6753. ['42, 130, 228', base.hexToRgba(temp.color)],
  6754. ['89, 125, 252', base.hexToRgba(temp.color)],
  6755. ]);
  6756. },
  6757. getToken() {
  6758. $doc.find('.loading-popup .loading-title').html(`令牌获取中`);
  6759. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取令牌~</div>`);
  6760. let token = base.getStorage("authorToken");
  6761. return token;
  6762. },
  6763. async getLink() {
  6764. Swal.fire({
  6765. ...temp.swalDefault,
  6766. showConfirmButton: false,
  6767. allowOutsideClick: false,
  6768. allowEscapeKey: false,
  6769. allowEnterKey: false,
  6770. title: "获取中",
  6771. html: `...`,
  6772. footer: "如果选的文件较多,请耐心等待获取完成哦!",
  6773. customClass: {
  6774. popup: 'loading-popup',
  6775. header: 'loading-header',
  6776. title: 'loading-title',
  6777. content: 'loading-content',
  6778. input: 'loading-input',
  6779. footer: 'loading-footer'
  6780. },
  6781. willOpen: function () {
  6782. Swal.showLoading();
  6783. },
  6784. });
  6785. let selectList = this.getSelectedList();
  6786. if (selectList.length === 0) return message.error('提示:<br/>请勾选要下载的文件哦~');
  6787. if (selectList.every(item => item.Type !== 0)) return message.error('提示:<br/>请打开文件夹后再勾选文件~');
  6788. if (temp.pege === 'home') {
  6789. let token = this.getToken();
  6790. let batchSize = 15;
  6791. let processed = 0;
  6792. selectList = selectList.filter(item => item.Type === 0);
  6793. for (let i = 0; i < selectList.length; i += batchSize) {
  6794. let batch = selectList.slice(i, i + batchSize);
  6795. let queue = [];
  6796. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  6797. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取文件对应的下载链接~</div>`);
  6798. batch.forEach((item, localIndex) => {
  6799. let globalIndex = i + localIndex;
  6800. queue.push(this.getFileUrlByOnce(item, globalIndex, token)
  6801. .then(val => {
  6802. processed++;
  6803. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  6804. return val;
  6805. }));
  6806. });
  6807. let res = await Promise.all(queue);
  6808. res.forEach(val => {
  6809. selectList[val.index].DownloadUrl = val.downloadUrl;
  6810. });
  6811. await base.sleep(1000);
  6812. }
  6813. temp.links = [selectList, {
  6814. isFolder: v => v.Type !== 0,
  6815. getFileName: v => v.FileName,
  6816. getFileSize: v => v.Size,
  6817. getFileLink: v => v.DownloadUrl || v.DownloadURL,
  6818. tooltip: config.$123pan.dom
  6819. }]
  6820. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  6821. } else if (temp.pege === 'share') {
  6822. let token = this.getToken();
  6823. let batchSize = 15;
  6824. let processed = 0;
  6825. selectList = selectList.filter(item => item.Type === 0);
  6826. let pathSplit = location.pathname.split('/').filter(Boolean);
  6827. let ShareKey = pathSplit[1];
  6828. for (let i = 0; i < selectList.length; i += batchSize) {
  6829. let batch = selectList.slice(i, i + batchSize);
  6830. let queue = [];
  6831. $doc.find('.loading-popup .loading-title').html(`链接获取中`);
  6832. $doc.find('.loading-popup .swal2-html-container').html(`<div>正在获取文件对应的下载链接~</div>`);
  6833. batch.forEach((item, localIndex) => {
  6834. let globalIndex = i + localIndex;
  6835. queue.push(this.getFileUrlByOnce(item, globalIndex, token, ShareKey)
  6836. .then(val => {
  6837. processed++;
  6838. $doc.find('.loading-popup .swal2-html-container').html(`<div>已获取 ${processed} / ${selectList.length} 个链接~</div>`);
  6839. return val;
  6840. }));
  6841. });
  6842. let res = await Promise.all(queue);
  6843. res.forEach(val => {
  6844. selectList[val.index].DownloadUrl = val.downloadUrl;
  6845. });
  6846. await base.sleep(1000);
  6847. }
  6848. temp.links = [selectList, {
  6849. isFolder: v => v.Type !== 0,
  6850. getFileName: v => v.FileName,
  6851. getFileSize: v => v.Size,
  6852. getFileLink: v => v.DownloadUrl || v.DownloadURL,
  6853. tooltip: config.$123pan.dom
  6854. }]
  6855. base.showMainDialog(config.base.dom.button[temp.mode].title, base.generateDom(temp.links), config.base.dom.button[temp.mode].footer);
  6856. } else {
  6857. return message.error('提示:<br/>页面错误~');
  6858. }
  6859. },
  6860. async getFileUrlByOnce(item, index, token, ShareKey) {
  6861. if (item.DownloadUrl || item.DownloadURL) {
  6862. let url = item.DownloadUrl ? item.DownloadUrl : item.DownloadURL;
  6863. let surl = new URL(url).searchParams.get("params");
  6864. if (surl) url = base.decodeBase(surl);
  6865. // url = await base.getFinalUrl(url);
  6866. return {
  6867. index,
  6868. downloadUrl: url
  6869. };
  6870. }
  6871. let res = null;
  6872. if (ShareKey) {
  6873. res = await base.post(config.$123pan.api.getShareLink, { "ShareKey": ShareKey, "FileID": item.FileId, "S3keyFlag": item.S3KeyFlag, "Size": item.Size, "Etag": item.Etag }, { "Authorization": `Bearer ${token}`, "Platform": "ios" });
  6874. } else {
  6875. res = await base.post(config.$123pan.api.getLink, { "driveId": 0, "etag": item.Etag, "fileId": item.FileId, "s3keyFlag": item.S3KeyFlag, "type": item.Type, "fileName": item.FileName, "size": item.Size }, { "Authorization": `Bearer ${token}`, "Platform": "ios" });
  6876. }
  6877. if (res.data?.DownloadUrl || res.data?.DownloadURL) {
  6878. let url = res.data.DownloadUrl ? res.data.DownloadUrl : res.data?.DownloadURL;
  6879. let surl = new URL(url).searchParams.get("params");
  6880. if (surl) url = base.decodeBase(surl);
  6881. // url = await base.getFinalUrl(url);
  6882. return {
  6883. index,
  6884. downloadUrl: url
  6885. };
  6886. } else if (res?.code === 5112) {
  6887. return message.error('提示:<br/>请先登录网盘后再获取链接呢~');
  6888. } else {
  6889. return {
  6890. index,
  6891. downloadUrl: '获取下载地址失败,刷新后再试试吧~'
  6892. };
  6893. }
  6894. },
  6895. getSelectedList() {
  6896. try {
  6897. let selectedList = [];
  6898. let reactDom = $(".ant-table-wrapper, .tiled-list, .file-list")[0];
  6899. let reactObj = base.findReact(reactDom);
  6900. let props = reactObj.pendingProps;
  6901. if (props) {
  6902. let fileList = props?.dataSource || props?.loadedFileList || props?.files || [];
  6903. fileList.forEach(function (val) {
  6904. if (val?.checked === true) {
  6905. selectedList.push(val);
  6906. }
  6907. });
  6908. }
  6909. return selectedList;
  6910. } catch (e) {
  6911. return [];
  6912. }
  6913. },
  6914. addButton() {
  6915. base.waitForKeyElements(config.$123pan.mount.home, (element) => {
  6916. temp.pege = $123pan.detectPage();
  6917. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'home') return;
  6918. let $button = $(`<button type="button" class="ant-btn ${[...document.querySelector('[class*="css-dev-only-do-not-override-"]').classList].find(c => /^css-dev-only-do-not-override-[a-z0-9]+$/.test(c))} ant-btn-primary ant-btn-color-primary ant-btn-variant-solid ant-dropdown-trigger mfy-button upload-button pl-button color-button" style="user-select: text !important;">
  6919. <svg class="icon home-operator-icon-upload" aria-hidden="true"><use xlink:href="#general_download_16_1"></use></svg>
  6920. <span>下载助手</span>
  6921. <ul class="pl-dropdown-menu" style="top:23px">
  6922. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  6923. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  6924. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  6925. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  6926. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  6927. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  6928. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  6929. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  6930. </ul>
  6931. </button>`);
  6932. element.prepend($button);
  6933. })
  6934. base.waitForKeyElements(config.$123pan.mount.share, (element) => {
  6935. element = element.parent();
  6936. temp.pege = $123pan.detectPage();
  6937. if ($(".pl-button").length > 0 || !temp.pege || temp.pege !== 'share') return;
  6938. let $button = $(`<div class="register pl-button color-button">
  6939. <svg class="icon" aria-hidden="true" style="color:rgb(255, 255, 255);margin-right:5px;"><use xlink:href="#top_btn_download2"></use></svg>下载助手
  6940. <ul class="pl-dropdown-menu" style="top:37px">
  6941. <li class="pl-button-mode" data-mode="api"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-downward"/></svg>API 下载</li>
  6942. <li class="pl-button-mode" data-mode="curl"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-plug"/></svg>cURL 下载</li>
  6943. <li class="pl-button-mode" data-mode="aria2"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>Aria2 下载</li>
  6944. <li class="pl-button-mode" data-mode="bitcomet"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>彗星下载</li>
  6945. <li class="pl-button-mode" data-mode="abdm"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-cloud-arrow-down"/></svg>ABDM 下载</li>
  6946. <li class="pl-button-mode listener-open-setting"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-gear"/></svg>助手设置</li>
  6947. <li class="pl-button-mode listener-open-beautify"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-palette"/></svg>助手美化</li>
  6948. <li class="pl-button-mode listener-open-updatelog"><svg class="pl-icon"><use xlink:href="#pl-icon-fa-newspaper"/></svg>更新日志</li>
  6949. </ul>
  6950. </div>`);
  6951. $button.css({ "width": "100px" })
  6952. element.append($button);
  6953. })
  6954. },
  6955. addInitButton() {
  6956. base.waitForKeyElements(config.$123pan.mount.home, (element) => {
  6957. temp.pege = $123pan.detectPage();
  6958. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'home') return;
  6959. let $button = $(`<button type="button" class="ant-btn ${[...document.querySelector('[class*="css-dev-only-do-not-override-"]').classList].find(c => /^css-dev-only-do-not-override-[a-z0-9]+$/.test(c))} ant-btn-primary ant-btn-color-primary ant-btn-variant-solid ant-dropdown-trigger mfy-button upload-button pl-button-init color-button" style="user-select: text !important;">
  6960. <svg class="icon home-operator-icon-upload" aria-hidden="true"><use xlink:href="#general_download_16_1"></use></svg>
  6961. <span>点我点亮</span>
  6962. </button>`);
  6963. $button.click(base.showInitDialog);
  6964. element.prepend($button);
  6965. })
  6966. base.waitForKeyElements(config.$123pan.mount.share, (element) => {
  6967. element = element.parent();
  6968. temp.pege = $123pan.detectPage();
  6969. if ($(".pl-button-init").length > 0 || !temp.pege || temp.pege !== 'share') return;
  6970. let $button = $(`<div class="register pl-button-init color-button">
  6971. <svg class="icon" aria-hidden="true" style="color:rgb(255, 255, 255);margin-right:5px;"><use xlink:href="#top_btn_download2"></use></svg>点我点亮
  6972. </div>`);
  6973. $button.click(base.showInitDialog);
  6974. $button.css({ "width": "100px" })
  6975. element.append($button);
  6976. })
  6977. },
  6978. detectPage() {
  6979. let path = location.pathname;
  6980. if (/^\/$/.test(path)) return 'home';
  6981. if (/^\/s\//.test(path)) return 'share';
  6982. return "";
  6983. },
  6984. async initPanLinker() {
  6985. base.createTip();
  6986. base.registerMenuCommand();
  6987. if (config.base.num === base.getValue('setting_init').code || config.base.license === base.getValue('setting_init').license) {
  6988. this.addButton();
  6989. } else {
  6990. this.addInitButton();
  6991. }
  6992. this.addPageListener();
  6993. },
  6994. };
  6995. // 主代码
  6996. let main = {
  6997. async init() {
  6998. base.waitForKeyElements(`html:not(:has(> .${mount})) head`, (element) => {
  6999. if ($(`.${mount}`).length > 0) return;
  7000. element.after(`<${mount} class="${mount}" />`);
  7001. })
  7002. base.waitForKeyElements(`.${mount}`, (element) => {
  7003. element.append(`<svg aria-hidden="true" style="position: absolute; width: 0px; height: 0px; overflow: hidden;">
  7004. <symbol id="pl-icon-fa-downward" viewBox="0 0 512 512">
  7005. <path d="M425.199,223.957c-13.303-13.303-34.961-13.303-48.205-0.06l-86.861,85.086V34.133C290.133,15.309,274.824,0,256,0 s-34.133,15.309-34.133,34.133v274.867l-86.801-85.052c-13.312-13.312-34.961-13.312-48.273,0 c-13.312,13.312-13.303,34.97,0,48.273c0.017,0.017,0.034,0.026,0.043,0.043l148.361,146.5c5.726,5.658,13.227,8.482,20.727,8.482 c7.543,0,15.078-2.859,20.787-8.568L425.199,272.23c6.451-6.443,10.001-15.019,10.001-24.132S431.65,230.409,425.199,223.957z"></path>
  7006. <path d="M401.067,443.733H110.933c-18.825,0-34.133,15.309-34.133,34.133S92.109,512,110.933,512h290.133 c18.825,0,34.133-15.309,34.133-34.133S419.883,443.733,401.067,443.733z"></path>
  7007. </symbol>
  7008. <symbol id="pl-icon-fa-plug" viewBox="0 0 384 512">
  7009. <path d="M96 0C78.3 0 64 14.3 64 32l0 96 64 0 0-96c0-17.7-14.3-32-32-32zM288 0c-17.7 0-32 14.3-32 32l0 96 64 0 0-96c0-17.7-14.3-32-32-32zM32 160c-17.7 0-32 14.3-32 32s14.3 32 32 32l0 32c0 77.4 55 142 128 156.8l0 67.2c0 17.7 14.3 32 32 32s32-14.3 32-32l0-67.2C297 398 352 333.4 352 256l0-32c17.7 0 32-14.3 32-32s-14.3-32-32-32L32 160z"></path>
  7010. </symbol>
  7011. <symbol id="pl-icon-fa-cloud-arrow-down" viewBox="0 0 640 512">
  7012. <path d="M144 480C64.5 480 0 415.5 0 336c0-62.8 40.2-116.2 96.2-135.9c-.1-2.7-.2-5.4-.2-8.1c0-88.4 71.6-160 160-160c59.3 0 111 32.2 138.7 80.2C409.9 102 428.3 96 448 96c53 0 96 43 96 96c0 12.2-2.3 23.8-6.4 34.6C596 238.4 640 290.1 640 352c0 70.7-57.3 128-128 128l-368 0zm79-167l80 80c9.4 9.4 24.6 9.4 33.9 0l80-80c9.4-9.4 9.4-24.6 0-33.9s-24.6-9.4-33.9 0l-39 39L344 184c0-13.3-10.7-24-24-24s-24 10.7-24 24l0 134.1-39-39c-9.4-9.4-24.6-9.4-33.9 0s-9.4 24.6 0 33.9z"></path>
  7013. </symbol>
  7014. <symbol id="pl-icon-fa-gear" viewBox="0 0 512 512">
  7015. <path d="M495.9 166.6c3.2 8.7 .5 18.4-6.4 24.6l-43.3 39.4c1.1 8.3 1.7 16.8 1.7 25.4s-.6 17.1-1.7 25.4l43.3 39.4c6.9 6.2 9.6 15.9 6.4 24.6c-4.4 11.9-9.7 23.3-15.8 34.3l-4.7 8.1c-6.6 11-14 21.4-22.1 31.2c-5.9 7.2-15.7 9.6-24.5 6.8l-55.7-17.7c-13.4 10.3-28.2 18.9-44 25.4l-12.5 57.1c-2 9.1-9 16.3-18.2 17.8c-13.8 2.3-28 3.5-42.5 3.5s-28.7-1.2-42.5-3.5c-9.2-1.5-16.2-8.7-18.2-17.8l-12.5-57.1c-15.8-6.5-30.6-15.1-44-25.4L83.1 425.9c-8.8 2.8-18.6 .3-24.5-6.8c-8.1-9.8-15.5-20.2-22.1-31.2l-4.7-8.1c-6.1-11-11.4-22.4-15.8-34.3c-3.2-8.7-.5-18.4 6.4-24.6l43.3-39.4C64.6 273.1 64 264.6 64 256s.6-17.1 1.7-25.4L22.4 191.2c-6.9-6.2-9.6-15.9-6.4-24.6c4.4-11.9 9.7-23.3 15.8-34.3l4.7-8.1c6.6-11 14-21.4 22.1-31.2c5.9-7.2 15.7-9.6 24.5-6.8l55.7 17.7c13.4-10.3 28.2-18.9 44-25.4l12.5-57.1c2-9.1 9-16.3 18.2-17.8C227.3 1.2 241.5 0 256 0s28.7 1.2 42.5 3.5c9.2 1.5 16.2 8.7 18.2 17.8l12.5 57.1c15.8 6.5 30.6 15.1 44 25.4l55.7-17.7c8.8-2.8 18.6-.3 24.5 6.8c8.1 9.8 15.5 20.2 22.1 31.2l4.7 8.1c6.1 11 11.4 22.4 15.8 34.3zM256 336a80 80 0 1 0 0-160 80 80 0 1 0 0 160z"></path>
  7016. </symbol>
  7017. <symbol id="pl-icon-fa-palette" viewBox="0 0 512 512">
  7018. <path d="M512 256c0 .9 0 1.8 0 2.7c-.4 36.5-33.6 61.3-70.1 61.3L344 320c-26.5 0-48 21.5-48 48c0 3.4 .4 6.7 1 9.9c2.1 10.2 6.5 20 10.8 29.9c6.1 13.8 12.1 27.5 12.1 42c0 31.8-21.6 60.7-53.4 62c-3.5 .1-7 .2-10.6 .2C114.6 512 0 397.4 0 256S114.6 0 256 0S512 114.6 512 256zM128 288a32 32 0 1 0 -64 0 32 32 0 1 0 64 0zm0-96a32 32 0 1 0 0-64 32 32 0 1 0 0 64zM288 96a32 32 0 1 0 -64 0 32 32 0 1 0 64 0zm96 96a32 32 0 1 0 0-64 32 32 0 1 0 0 64z"></path>
  7019. </symbol>
  7020. <symbol id="pl-icon-fa-newspaper" viewBox="0 0 512 512">
  7021. <path d="M96 96c0-35.3 28.7-64 64-64l288 0c35.3 0 64 28.7 64 64l0 320c0 35.3-28.7 64-64 64L80 480c-44.2 0-80-35.8-80-80L0 128c0-17.7 14.3-32 32-32s32 14.3 32 32l0 272c0 8.8 7.2 16 16 16s16-7.2 16-16L96 96zm64 24l0 80c0 13.3 10.7 24 24 24l112 0c13.3 0 24-10.7 24-24l0-80c0-13.3-10.7-24-24-24L184 96c-13.3 0-24 10.7-24 24zm208-8c0 8.8 7.2 16 16 16l48 0c8.8 0 16-7.2 16-16s-7.2-16-16-16l-48 0c-8.8 0-16 7.2-16 16zm0 96c0 8.8 7.2 16 16 16l48 0c8.8 0 16-7.2 16-16s-7.2-16-16-16l-48 0c-8.8 0-16 7.2-16 16zM160 304c0 8.8 7.2 16 16 16l256 0c8.8 0 16-7.2 16-16s-7.2-16-16-16l-256 0c-8.8 0-16 7.2-16 16zm0 96c0 8.8 7.2 16 16 16l256 0c8.8 0 16-7.2 16-16s-7.2-16-16-16l-256 0c-8.8 0-16 7.2-16 16z"></path>
  7022. </symbol>
  7023. <symbol id="pl-icon-fa-cloud-arrow-up" viewBox="0 0 640 512">
  7024. <path d="M144 480C64.5 480 0 415.5 0 336c0-62.8 40.2-116.2 96.2-135.9c-.1-2.7-.2-5.4-.2-8.1c0-88.4 71.6-160 160-160c59.3 0 111 32.2 138.7 80.2C409.9 102 428.3 96 448 96c53 0 96 43 96 96c0 12.2-2.3 23.8-6.4 34.6C596 238.4 640 290.1 640 352c0 70.7-57.3 128-128 128l-368 0zm79-217c-9.4 9.4-9.4 24.6 0 33.9s24.6 9.4 33.9 0l39-39L296 392c0 13.3 10.7 24 24 24s24-10.7 24-24l0-134.1 39 39c9.4 9.4 24.6 9.4 33.9 0s9.4-24.6 0-33.9l-80-80c-9.4-9.4-24.6-9.4-33.9 0l-80 80z"></path>
  7025. </symbol>
  7026. <symbol id="pl-icon-fa-copy" viewBox="0 0 448 512">
  7027. <path d="M208 0L332.1 0c12.7 0 24.9 5.1 33.9 14.1l67.9 67.9c9 9 14.1 21.2 14.1 33.9L448 336c0 26.5-21.5 48-48 48l-192 0c-26.5 0-48-21.5-48-48l0-288c0-26.5 21.5-48 48-48zM48 128l80 0 0 64-64 0 0 256 192 0 0-32 64 0 0 48c0 26.5-21.5 48-48 48L48 512c-26.5 0-48-21.5-48-48L0 176c0-26.5 21.5-48 48-48z"></path>
  7028. </symbol>
  7029. <symbol id="pl-icon-fa-check" viewBox="0 0 448 512">
  7030. <path d="M438.6 105.4c12.5 12.5 12.5 32.8 0 45.3l-256 256c-12.5 12.5-32.8 12.5-45.3 0l-128-128c-12.5-12.5-12.5-32.8 0-45.3s32.8-12.5 45.3 0L160 338.7 393.4 105.4c12.5-12.5 32.8-12.5 45.3 0z"></path>
  7031. </symbol>
  7032. <symbol id="pl-icon-fa-list-check" viewBox="0 0 512 512">
  7033. <path d="M152.1 38.2c9.9 8.9 10.7 24 1.8 33.9l-72 80c-4.4 4.9-10.6 7.8-17.2 7.9s-12.9-2.4-17.6-7L7 113C-2.3 103.6-2.3 88.4 7 79s24.6-9.4 33.9 0l22.1 22.1 55.1-61.2c8.9-9.9 24-10.7 33.9-1.8zm0 160c9.9 8.9 10.7 24 1.8 33.9l-72 80c-4.4 4.9-10.6 7.8-17.2 7.9s-12.9-2.4-17.6-7L7 273c-9.4-9.4-9.4-24.6 0-33.9s24.6-9.4 33.9 0l22.1 22.1 55.1-61.2c8.9-9.9 24-10.7 33.9-1.8zM224 96c0-17.7 14.3-32 32-32l224 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-224 0c-17.7 0-32-14.3-32-32zm0 160c0-17.7 14.3-32 32-32l224 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-224 0c-17.7 0-32-14.3-32-32zM160 416c0-17.7 14.3-32 32-32l288 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-288 0c-17.7 0-32-14.3-32-32zM48 368a48 48 0 1 1 0 96 48 48 0 1 1 0-96z"></path>
  7034. </symbol>
  7035. <symbol id="pl-icon-fa-x-mark" viewBox="0 0 384 512">
  7036. <path d="M342.6 150.6c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L192 210.7 86.6 105.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3L146.7 256 41.4 361.4c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0L192 301.3 297.4 406.6c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L237.3 256 342.6 150.6z"></path>
  7037. </symbol>
  7038. <symbol id="pl-icon-fa-unlock-keyhole" viewBox="0 0 448 512">
  7039. <path d="M224 64c-44.2 0-80 35.8-80 80l0 48 240 0c35.3 0 64 28.7 64 64l0 192c0 35.3-28.7 64-64 64L64 512c-35.3 0-64-28.7-64-64L0 256c0-35.3 28.7-64 64-64l16 0 0-48C80 64.5 144.5 0 224 0c57.5 0 107 33.7 130.1 82.3c7.6 16 .8 35.1-15.2 42.6s-35.1 .8-42.6-15.2C283.4 82.6 255.9 64 224 64zm32 320c17.7 0 32-14.3 32-32s-14.3-32-32-32l-64 0c-17.7 0-32 14.3-32 32s14.3 32 32 32l64 0z"></path>
  7040. </symbol>
  7041. <symbol id="pl-icon-fa-star" viewBox="0 0 576 512">
  7042. <path d="M316.9 18C311.6 7 300.4 0 288.1 0s-23.4 7-28.8 18L195 150.3 51.4 171.5c-12 1.8-22 10.2-25.7 21.7s-.7 24.2 7.9 32.7L137.8 329 113.2 474.7c-2 12 3 24.2 12.9 31.3s23 8 33.8 2.3l128.3-68.5 128.3 68.5c10.8 5.7 23.9 4.9 33.8-2.3s14.9-19.3 12.9-31.3L438.5 329 542.7 225.9c8.6-8.5 11.7-21.2 7.9-32.7s-13.7-19.9-25.7-21.7L381.2 150.3 316.9 18z"></path>
  7043. </symbol>
  7044. <symbol id="pl-icon-fa-link" viewBox="0 0 640 512">
  7045. <path d="M579.8 267.7c56.5-56.5 56.5-148 0-204.5c-50-50-128.8-56.5-186.3-15.4l-1.6 1.1c-14.4 10.3-17.7 30.3-7.4 44.6s30.3 17.7 44.6 7.4l1.6-1.1c32.1-22.9 76-19.3 103.8 8.6c31.5 31.5 31.5 82.5 0 114L422.3 334.8c-31.5 31.5-82.5 31.5-114 0c-27.9-27.9-31.5-71.8-8.6-103.8l1.1-1.6c10.3-14.4 6.9-34.4-7.4-44.6s-34.4-6.9-44.6 7.4l-1.1 1.6C206.5 251.2 213 330 263 380c56.5 56.5 148 56.5 204.5 0L579.8 267.7zM60.2 244.3c-56.5 56.5-56.5 148 0 204.5c50 50 128.8 56.5 186.3 15.4l1.6-1.1c14.4-10.3 17.7-30.3 7.4-44.6s-30.3-17.7-44.6-7.4l-1.6 1.1c-32.1 22.9-76 19.3-103.8-8.6C74 372 74 321 105.5 289.5L217.7 177.2c31.5-31.5 82.5-31.5 114 0c27.9 27.9 31.5 71.8 8.6 103.9l-1.1 1.6c-10.3 14.4-6.9 34.4 7.4 44.6s34.4 6.9 44.6-7.4l1.1-1.6C433.5 260.8 427 182 377 132c-56.5-56.5-148-56.5-204.5 0L60.2 244.3z"></path>
  7046. </symbol>
  7047. <symbol id="pl-icon-si-tampermonkey" viewBox="0 0 24 24">
  7048. <path d="M5.955.002C3-.071.275 2.386.043 5.335c-.069 3.32-.011 6.646-.03 9.969.06 1.87-.276 3.873.715 5.573 1.083 2.076 3.456 3.288 5.77 3.105 4.003-.011 8.008.022 12.011-.017 2.953-.156 5.478-2.815 5.482-5.772-.007-4.235.023-8.473-.015-12.708C23.82 2.533 21.16.007 18.205.003c-4.083-.005-8.167 0-12.25-.002zm.447 12.683c2.333-.046 4.506 1.805 4.83 4.116.412 2.287-1.056 4.716-3.274 5.411-2.187.783-4.825-.268-5.874-2.341-1.137-2.039-.52-4.827 1.37-6.197a4.896 4.896 0 012.948-.99zm11.245 0c2.333-.046 4.505 1.805 4.829 4.116.413 2.287-1.056 4.716-3.273 5.411-2.188.783-4.825-.268-5.875-2.341-1.136-2.039-.52-4.827 1.37-6.197a4.896 4.896 0 012.949-.99z"/>
  7049. </symbol>
  7050. </svg>`);
  7051. return true;
  7052. }, true)
  7053. // 智能默认设置
  7054. base.initDefaultConfig();
  7055. // 加载美化样式
  7056. base.addPanLinkerStyle();
  7057. // 加载按钮监听
  7058. base.addPageListener();
  7059. // 创建下载用 iframe
  7060. base.createDownloadIframe();
  7061. /**
  7062. * 控制台输出
  7063. * @author 油小猴
  7064. * @author hmjz100
  7065. * @description 来自【网盘智能识别助手】,有改动
  7066. */
  7067. console.log(`%c %c LinkSwift\n一个基于 JavaScript 的网盘文件下载地址获取工具\n仓库:https://github.com/hmjz100/LinkSwift\n版本:${info.version}\n领域:${(window.self !== window.top ? "[iframe] " : "") + (document.title ? (document.title + " (" + location.origin + location.pathname + ")") : location.href)}`, `background:url(${info.icon}) center center no-repeat;background-size:12px;padding:3px`, `padding:2px`);
  7068. // 最后判断页面地址并加载对应的initPanLinker
  7069. if (/(pan|yun).baidu.com/.test(location.host)) {
  7070. base.getValue('setting_ui_theme').custom.$baidu === true && $baidu.beautifyPage();
  7071. $baidu.initPanLinker();
  7072. $baidu.greenerPage();
  7073. }
  7074. if (/openapi.baidu.com\/oauth/.test(location.href)) {
  7075. $baidu.initAuthorize()
  7076. }
  7077. if (/www.(aliyundrive|alipan).com/.test(location.host)) {
  7078. base.getValue('setting_ui_theme').custom.$aliyun === true && $aliyun.beautifyPage();
  7079. $aliyun.initPanLinker();
  7080. $aliyun.greenerPage();
  7081. }
  7082. if (/(yun|caiyun).139.com/.test(location.host)) {
  7083. base.getValue('setting_ui_theme').custom.$mcloud === true && $mcloud.beautifyPage();
  7084. $mcloud.initPanLinker();
  7085. $mcloud.greenerPage();
  7086. }
  7087. if (/cloud.189.cn/.test(location.host)) {
  7088. base.getValue('setting_ui_theme').custom.$tcloud === true && $tcloud.beautifyPage();
  7089. $tcloud.initPanLinker();
  7090. $tcloud.greenerPage();
  7091. }
  7092. if (/pan.xunlei.com/.test(location.host)) {
  7093. base.getValue('setting_ui_theme').custom.$xunlei === true && $xunlei.beautifyPage();
  7094. $xunlei.initPanLinker();
  7095. }
  7096. if (/pan.quark.cn/.test(location.host)) {
  7097. base.getValue('setting_ui_theme').custom.$quark === true && $quark.beautifyPage();
  7098. $quark.initPanLinker();
  7099. $quark.greenerPage();
  7100. }
  7101. if (/drive.uc.cn/.test(location.host)) {
  7102. base.getValue('setting_ui_theme').custom.$uc === true && $uc.beautifyPage();
  7103. $uc.initPanLinker();
  7104. $uc.greenerPage();
  7105. }
  7106. if (/(www|login).(123(pan|684|865|952|912).com|123pan.cn)/.test(location.host)) {
  7107. base.getValue('setting_ui_theme').custom.$123pan === true && $123pan.beautifyPage();
  7108. $123pan.initPanLinker();
  7109. $123pan.greenerPage();
  7110. }
  7111. let storedVersion = base.getValue("setting_init").version;
  7112. if (!storedVersion || base.isNewerVersion(info.version, storedVersion)) {
  7113. base.waitForKeyElements("body:not(.swal2-shown)", async () => {
  7114. await base.showUpdate();
  7115. let list = base.getValue("setting_init");
  7116. list.version = info.version;
  7117. base.setValue("setting_init", list);
  7118. return true;
  7119. }, true);
  7120. }
  7121. }
  7122. };
  7123. main.init();
  7124. // 这是啥?我不到啊
  7125. function idontknow(input) {
  7126. let charArray = input.split("");
  7127. // Fisher-Yates 洗牌算法
  7128. for (let i = charArray.length - 1; i > 0; i--) {
  7129. let j = Math.floor(Math.random() * (i + 1));
  7130. [charArray[i], charArray[j]] = [charArray[j], charArray[i]];
  7131. }
  7132. return charArray.join("");
  7133. }
  7134. })();