Greasy Fork is available in English.

网盘链接识别

识别网页中显示的网盘链接,目前包括百度网盘、蓝奏云、天翼云、中国移动云盘(原:和彩云)、阿里云、文叔叔、奶牛快传、123盘、腾讯微云、迅雷网盘、115网盘、夸克网盘、城通网盘(部分)、坚果云、UC网盘、BT磁力,支持蓝奏云、天翼云(需登录)、123盘、奶牛、UC网盘(需登录)、坚果云(需登录)和阿里云盘(需登录,且限制在网盘页面解析)直链获取下载,页面动态监控加载的链接,可自定义规则来识别小众网盘/网赚网盘或其它自定义的链接。

// ==UserScript==
// @name         网盘链接识别
// @namespace    https://greasyfork.org/zh-CN/scripts/445489
// @supportURL   https://github.com/WhiteSevs/TamperMonkeyScript/issues
// @icon         
// @version      2024.4.15.16
// @description  识别网页中显示的网盘链接,目前包括百度网盘、蓝奏云、天翼云、中国移动云盘(原:和彩云)、阿里云、文叔叔、奶牛快传、123盘、腾讯微云、迅雷网盘、115网盘、夸克网盘、城通网盘(部分)、坚果云、UC网盘、BT磁力,支持蓝奏云、天翼云(需登录)、123盘、奶牛、UC网盘(需登录)、坚果云(需登录)和阿里云盘(需登录,且限制在网盘页面解析)直链获取下载,页面动态监控加载的链接,可自定义规则来识别小众网盘/网赚网盘或其它自定义的链接。
// @author       WhiteSevs
// @match        *://*/*
// @run-at       document-end
// @license      GPL-3.0-only
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_xmlhttpRequest
// @grant        GM_download
// @grant        GM_addStyle
// @grant        GM_info
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        unsafeWindow
// @connect      *
// @connect      lanzoub.com
// @connect      lanzouc.com
// @connect      lanzoue.com
// @connect      lanzouf.com
// @connect      lanzoug.com
// @connect      lanzouh.com
// @connect      lanzoui.com
// @connect      lanzouj.com
// @connect      lanzouk.com
// @connect      lanzoul.com
// @connect      lanzoum.com
// @connect      lanzouo.com
// @connect      lanzoup.com
// @connect      lanzouq.com
// @connect      lanzous.com
// @connect      lanzout.com
// @connect      lanzouu.com
// @connect      lanzouv.com
// @connect      lanzouw.com
// @connect      lanzoux.com
// @connect      lanzouy.com
// @connect      lanosso.com
// @connect      lanzn.com
// @connect      lanzog.com
// @connect      lanpw.com
// @connect      lanpv.com
// @connect      lanzv.com
// @connect      wwentua.com
// @connect      ilanzou.com
// @connect      189.cn
// @connect      123pan.com
// @connect      123pan.cn
// @connect      wenshushu.cn
// @connect      jianguoyun.com
// @connect      cowtransfer.com
// @connect      cowcs.com
// @connect      aliyundrive.com
// @connect      baidu.com
// @connect      139.com
// @connect      weiyun.com
// @connect      xunlei.com
// @connect      115.com
// @connect      quark.cn
// @connect      jianguoyun.com
// @connect      uc.cn
// @connect      ctfile.com
// @connect      sharepoint.com
// @exclude      /^http(s|):\/\/s1\.hdslb\.com\/.*$/
// @exclude      /^http(s|):\/\/www\.bilibili\.com\/video.*$/
// @exclude      /^http(s|):\/\/message\.bilibili\.com\/.*$/
// @exclude      /^http(s|):\/\/live\.bilibili\.com\/.*$/
// @exclude      /^http(s|):\/\/.*\.mail\.qq\.com\/.*$/
// @exclude      /^http(s|):\/\/.*video\.qq\.com\/.*$/
// @exclude      /^http(s|):\/\/.*\.vscode-cdn\.net\/.*$/
// @exclude      /^http(s|):\/\/.*vscode\.dev\/.*$/
// @exclude      /^http(s|):\/\/cloudgame\.ds\.163\.com\/.*$/
// @exclude      /^http(s|):\/\/cloudgame\.webapp\.163\.com\/.*$/
// @exclude      /^http(s|):\/\/cg\.163\.com\/.*$/
// @require      https://update.greasyfork.org/scripts/462234/1322684/Message.js
// @require      https://update.greasyfork.org/scripts/456470/1320377/%E7%BD%91%E7%9B%98%E9%93%BE%E6%8E%A5%E8%AF%86%E5%88%AB-%E5%9B%BE%E6%A0%87%E5%BA%93.js
// @require      https://update.greasyfork.org/scripts/465550/1360573/JS-%E5%88%86%E9%A1%B5%E6%8F%92%E4%BB%B6.js
// @require      https://update.greasyfork.org/scripts/456485/1360571/pops.js
// @require      https://update.greasyfork.org/scripts/455186/1360586/WhiteSevsUtils.js
// @require      https://update.greasyfork.org/scripts/465772/1360574/DOMUtils.js
// @require      https://update.greasyfork.org/scripts/486152/1360580/Crypto-JS.js
// ==/UserScript==

(function () {
  if (typeof unsafeWindow === "undefined") {
    unsafeWindow = globalThis || window;
  }
  /**
   * @type {import("../库/CryptoJS/index")}
   */
  const Cryptojs = CryptoJS ?? window.CryptoJS ?? unsafeWindow.CryptoJS;
  /**
   * @type {import("../库/Qmsg")}
   */
  const Qmsg = window.Qmsg;
  /**
   * @type {import("../库/pops")}
   */
  const pops = window.pops;
  const AnyTouch = pops.config.Utils.AnyTouch();
  /** @type {Utils} */
  const utils = window.Utils.noConflict();
  /** @type {DOMUtils} */
  const DOMUtils = window.DOMUtils.noConflict();
  const log = new utils.Log(GM_info, unsafeWindow.console || console);
  const httpx = new utils.Httpx(GM_xmlhttpRequest);
  const GM_Menu = new utils.GM_Menu({
    autoReload: false,
    GM_getValue,
    GM_setValue,
    GM_registerMenuCommand,
    GM_unregisterMenuCommand,
  });

  /**
   * 数据配置
   */
  const NetDiskData = {
    /** 蓝奏云默认主机域名 */
    lanzou_defaultHostName: "www.lanzout.com",
    /** 是否点击网盘小图标寻找shareCode */
    iconDefaultClickEventToFindShareCode: true,
    /** 是否点击网盘小图标寻找shareCode且使用光标选中 */
    iconDefaultClickEventToFindShareCodeWithSelect: true,
    /** 是否点击网盘小图标(循环)寻找shareCode */
    iconDefaultClickEventToFindShareCodeByLoop: true,
    /** 匹配间隔默认时间(s) */
    defaultMatchIntervalTime: 0.8,
    /** 匹配间隔最小时间(s) */
    matchIntervalMinTime: 0.0,
    /** 匹配间隔最打时间(s) */
    matchIntervalMaxTime: 5.0,

    /** 小窗的默认值-宽度 */
    defaultSmallWindowWidth: 250,
    /** 小窗的最小值-宽度 */
    smallWindowMinWidth: 50,
    /** 小窗的最大值-宽度 */
    smallWindowMaxWidth: DOMUtils.width(globalThis),

    /** 小窗的默认值-最大高度 */
    defaultSmallWindowMaxHeight: 200,
    /** 小窗的最小值-最大高度 */
    smallWindowMinMaxHeight: 50,
    /** 小窗的最大值-最大高度 */
    smallWindowMaxMaxHeight: DOMUtils.height(globalThis),

    /** 相同字符系数-默认值 */
    defaultExcludeIdenticalSharedCodesCoefficient: 1,
    /** 相同字符系数-最小值 */
    excludeIdenticalSharedCodesCoefficientMin: 0,
    /** 相同字符系数-最大值 */
    excludeIdenticalSharedCodesCoefficientMax: 1,

    /** 行为模式-默认值:悬浮按钮+小窗 */
    defaultNetdiskBehaviorMode: "suspension_smallwindow",
  };

  const NetDiskUIMenuData = {
    /**
     * @returns {string}
     */
    get netdiskBehaviorMode() {
      return GM_getValue(
        "netdisk-behavior-mode",
        NetDiskData.defaultNetdiskBehaviorMode
      ).toLowerCase();
    },
  };

  const NetDiskLocalData = {
    /**
     * 是否启用该规则
     * @param {string} name
     * @param {boolean} defaultValue
     * @returns
     */
    enable(name, defaultValue = true) {
      return Boolean(GM_getValue(`${name}-enable`, defaultValue));
    },
    /**
     * 提取码间隔前的字符长度
     * @param {string} name
     * @param {number} defaultValue 默认值: 20
     */
    innerTextAccessCodeBeforeMaxRange(name, defaultValue = 20) {
      return parseInt(
        GM_getValue(
          `${name}_innerText_accessCode_before_max_range`,
          defaultValue
        )
      );
    },
    /**
     * 提取码间隔后的字符长度
     * @param {string} name
     * @param {number} defaultValue 默认值: 10
     */
    innerTextAccessCodeAfterMaxRange(name, defaultValue = 10) {
      return parseInt(
        GM_getValue(
          `${name}_innerText_accessCode_after_max_range`,
          defaultValue
        )
      );
    },
    /**
     * 提取码间隔前的字符长度
     * @param {string} name
     * @param {number} defaultValue 默认值: 100
     */
    innerHTMLAccessCodeBeforeMaxRange(name, defaultValue = 100) {
      return parseInt(
        GM_getValue(
          `${name}_innerHTML_accessCode_before_max_range`,
          defaultValue
        )
      );
    },
    /**
     * 提取码间隔后的字符长度
     * @param {string} name
     * @param {number} defaultValue 默认值: 15
     */
    innerHTMLAccessCodeAfterMaxRange(name, defaultValue = 15) {
      return parseInt(
        GM_getValue(
          `${name}_innerHTML_accessCode_after_max_range`,
          defaultValue
        )
      );
    },
    /**
     * 是否进行校验链接有效性
     * @param {string} name
     * @param {boolean} [defaultValue=false]
     * @returns
     */
    checkLinkValidity(name, defaultValue = false) {
      return Boolean(GM_getValue(`${name}-check-link-valid`, defaultValue));
    },
  };

  const NetDisk = {
    /**
     * 是否初始化
     * @type {boolean}
     */
    isInit: false,
    /**
     * 链接字典,识别规则->识别到的访问码|分享码|下标
     * @type {UtilsDictionaryConstructor<string,UtilsDictionaryConstructor<string,NetDiskDictData>>}
     */
    linkDict: void 0,
    /**
     * (临时)链接字典
     * @type {UtilsDictionaryConstructor<string,UtilsDictionaryConstructor<string,NetiDiskHandleObject>>}
     */
    tempLinkDict: void 0,
    /**
     * 是否正在匹配链接中
     */
    isMatching: false,
    /**
     * 用于存储已匹配到的网盘规则名
     * 只有单独的名
     * @type {Set<string>}
     */
    matchLink: void 0,
    /**
     * 是否成功匹配到链接
     */
    hasMatchLink: false,
    /**
     * 剪贴板内容
     */
    clipboardText: "",
    /**
     * 是否允许匹配window.location.href
     */
    allowMatchLocationHref: false,
    /**
     * 使用该正则判断提取到的shareCode是否正确
     */
    shareCodeNotMatchRegexpList: [
      /(vipstyle|notexist|ajax|file|download|ptqrshow|xy-privacy|comp|web|undefined|1125|unproved|console|account|favicon)/g,
    ],
    /**
     * 使用该正则判断提取到的accessCode是否正确
     */
    accessCodeNotMatchRegexpList: [/^(font)/gi],
    /**
     * 当没有accessCode时,使用该正则去除不需要的字符串
     */
    noAccessCodeRegExp:
      /( |提取码:|\n密码:{#accessCode#}|{#accessCode#}|{#encodeURI-accessCode#}|{#encodeURIComponent-accessCode#}|{#decodeURI-accessCode#}|{#encodeURIComponent-accessCode#}|\?pwd=|&pwd=)/gi,
    /**
     * @type {NetDiskRegular}
     */
    regular: {
      baidu: [
        {
          enable: NetDiskLocalData.enable("baidu"),
          link_innerText: `pan.baidu.com/s/[0-9a-zA-Z-_]{6,24}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "baidu"
          )}}(密码|访问码|提取码|\\?pwd=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "baidu"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `pan.baidu.com/s/[0-9a-zA-Z-_]{6,24}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "baidu"
          )}}(密码|访问码|提取码|\\?pwd=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "baidu"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /pan\.baidu\.com\/s\/([0-9a-zA-Z-_]+)/gi,
          shareCodeNeedRemoveStr: /pan\.baidu\.com\/s\//gi,
          checkAccessCode: /(密码|访问码|提取码|pwd=)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "pan.baidu.com/s/{#shareCode#}?pwd={#accessCode#}",
          blank: "https://pan.baidu.com/s/{#shareCode#}?pwd={#accessCode#}",
          copyUrl: "https://pan.baidu.com/s/{#shareCode#}?pwd={#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("baidu"),
        },
        {
          enable: NetDiskLocalData.enable("baidu"),
          link_innerText: `pan.baidu.com/(share|wap)/init\\?surl=[0-9a-zA-Z-_]{5,24}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "baidu"
          )}}(密码|访问码|提取码|&pwd=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "baidu"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `pan.baidu.com/(share|wap)/init\\?surl=[0-9a-zA-Z-_]{5,24}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "baidu"
          )}}(密码|访问码|提取码|&pwd=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "baidu"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode:
            /pan\.baidu\.com\/(share|wap)\/init\?surl=([0-9a-zA-Z-_]+)/gi,
          shareCodeNeedRemoveStr: /pan\.baidu\.com\/(share|wap)\/init\?surl=/gi,
          checkAccessCode: /(密码|访问码|提取码|&pwd=)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow:
            "pan.baidu.com/share/init?surl={#shareCode#}&pwd={#accessCode#}",
          blank:
            "https://pan.baidu.com/share/init?surl={#shareCode#}&pwd={#accessCode#}",
          copyUrl:
            "https://pan.baidu.com/share/init?surl={#shareCode#}&pwd={#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("baidu"),
        },
      ],
      lanzou: [
        {
          enable: NetDiskLocalData.enable("lanzou"),
          link_innerText: `(lanzou[a-z]{0,1}|lan[a-z]{2}).com/(tp/|u/|)([a-zA-Z0-9_-]{5,22}|[%0-9a-zA-Z]{4,90}|[\\u4e00-\\u9fa5]{1,20})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "lanzou"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "lanzou"
          )}}[a-zA-Z0-9]{3,6}|)`,
          link_innerHTML: `(lanzou[a-z]{0,1}|lan[a-z]{2}).com/(tp/|u/|)([a-zA-Z0-9_-]{5,22}|[%0-9a-zA-Z]{4,90}|[\\u4e00-\\u9fa5]{1,20})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "lanzou"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "lanzou"
          )}}[a-zA-Z0-9]{3,6}|)`,
          shareCode:
            /(lanzou[a-z]{0,1}|lan[a-z]{2}).com\/(tp\/|u\/|)([a-zA-Z0-9_\-]{5,22}|[%0-9a-zA-Z]{4,90}|[\u4e00-\u9fa5]{1,20})/gi,
          shareCodeNeedRemoveStr:
            /(lanzou[a-z]{0,1}|lan[a-z]{2}).com\/(tp\/|u\/|)/gi,
          shareCodeExcludeRegular: ["lanzouyx"],
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{3,})/gi,
          uiLinkShow: `${GM_getValue(
            "lanzou-host-name",
            NetDiskData.lanzou_defaultHostName
          )}/{#shareCode#} 提取码: {#accessCode#}`,
          blank: `https://${GM_getValue(
            "lanzou-host-name",
            NetDiskData.lanzou_defaultHostName
          )}/{#shareCode#}`,
          copyUrl: `https://${GM_getValue(
            "lanzou-host-name",
            NetDiskData.lanzou_defaultHostName
          )}/{#shareCode#}\n密码:{#accessCode#}`,
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("lanzou"),
        },
        {
          enable: NetDiskLocalData.enable("lanzou"),
          link_innerText: `(lanzou[a-z]{0,1}|lan[a-z]{2}).com/s/([a-zA-Z0-9_-]{5,22}|[%0-9a-zA-Z]{4,90}|[\\u4e00-\\u9fa5]{1,20})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "lanzou"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "lanzou"
          )}}[a-zA-Z0-9]{3,6}|)`,
          link_innerHTML: `(lanzou[a-z]{0,1}|lan[a-z]{2}).com/s/([a-zA-Z0-9_-]{5,22}|[%0-9a-zA-Z]{4,90}|[\\u4e00-\\u9fa5]{1,20})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "lanzou"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "lanzou"
          )}}[a-zA-Z0-9]{3,6}|)`,
          shareCode:
            /(lanzou[a-z]{0,1}|lan[a-z]{2}).com\/s\/([a-zA-Z0-9_\-]{5,22}|[%0-9a-zA-Z]{4,90}|[\u4e00-\u9fa5]{1,20})/gi,
          shareCodeNeedRemoveStr: /(lanzou[a-z]{0,1}|lan[a-z]{2}).com\/s\//gi,
          shareCodeExcludeRegular: ["lanzouyx"],
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{3,})/gi,
          uiLinkShow: `${GM_getValue(
            "lanzou-host-name",
            NetDiskData.lanzou_defaultHostName
          )}/s/{#shareCode#} 提取码: {#accessCode#}`,
          blank: `https://${GM_getValue(
            "lanzou-host-name",
            NetDiskData.lanzou_defaultHostName
          )}/s/{#shareCode#}`,
          copyUrl: `https://${GM_getValue(
            "lanzou-host-name",
            NetDiskData.lanzou_defaultHostName
          )}/s/{#shareCode#}\n密码:{#accessCode#}`,
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("lanzou"),
        },
      ],
      lanzouyx: [
        {
          enable: NetDiskLocalData.enable("lanzouyx"),
          link_innerText: `ilanzou.com/s/([a-zA-Z0-9_-]{5,22})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "lanzouyx"
          )}}(密码|访问码|提取码|\\?code=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "lanzouyx"
          )}}[a-zA-Z0-9]{3,6}|)`,
          link_innerHTML: `ilanzou.com/s/([a-zA-Z0-9_-]{5,22})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "lanzouyx"
          )}}(密码|访问码|提取码|\\?code=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "lanzouyx"
          )}}[a-zA-Z0-9]{3,6}|)`,
          shareCode: /ilanzou.com\/s\/([a-zA-Z0-9_\-]{5,22})/gi,
          shareCodeNeedRemoveStr: /ilanzou.com\/s\//gi,
          checkAccessCode: /(密码|访问码|提取码|\?code=)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{3,})/gi,
          uiLinkShow: `www.ilanzou.com/s/{#shareCode#} 提取码: {#accessCode#}`,
          blank: `https://www.ilanzou.com/s/{#shareCode#}?code={#accessCode#}`,
          copyUrl: `https://www.ilanzou.com/s/{#shareCode#}?code={#accessCode#}`,
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("lanzouyx"),
        },
      ],
      tianyiyun: [
        {
          enable: NetDiskLocalData.enable("tianyiyun"),
          link_innerText: `(cloud.189.cn/web/share\\?code=([0-9a-zA-Z_-]){8,14}|cloud.189.cn/t/([a-zA-Z0-9_-]{8,14}))([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "tianyiyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "tianyiyun"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `(cloud.189.cn/web/share\\?code=([0-9a-zA-Z_\-]){8,14}|cloud.189.cn/t/([a-zA-Z0-9_-]{8,14}))([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "tianyiyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "tianyiyun"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode:
            /cloud.189.cn\/web\/share\?code=([0-9a-zA-Z_\-]){8,14}|cloud.189.cn\/t\/([a-zA-Z0-9_\-]{8,14})/gi,
          shareCodeNeedRemoveStr:
            /cloud\.189\.cn\/t\/|cloud.189.cn\/web\/share\?code=/gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "cloud.189.cn/t/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://cloud.189.cn/t/{#shareCode#}",
          copyUrl: "https://cloud.189.cn/t/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("tianyiyun"),
        },
      ],
      hecaiyun: [
        {
          enable: NetDiskLocalData.enable("hecaiyun"),
          link_innerText: `caiyun.139.com/m/i\\?([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "hecaiyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "hecaiyun"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `caiyun.139.com/m/i\\?([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "hecaiyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "hecaiyun"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /caiyun\.139\.com\/m\/i\?([a-zA-Z0-9_\-]{8,14})/gi,
          shareCodeNeedRemoveStr: /caiyun\.139\.com\/m\/i\?/gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "caiyun.139.com/m/i?{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://caiyun.139.com/m/i?{#shareCode#}",
          copyUrl:
            "https://caiyun.139.com/m/i?{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("hecaiyun"),
        },
        {
          enable: NetDiskLocalData.enable("hecaiyun"),
          link_innerText: `yun.139.com/link/w/i/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "hecaiyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "hecaiyun"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `yun.139.com/link/w/i/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "hecaiyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "hecaiyun"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /yun\.139\.com\/link\/w\/i\/([a-zA-Z0-9_\-]{8,14})/gi,
          shareCodeNeedRemoveStr: /yun\.139\.com\/link\/w\/i\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow:
            "yun.139.com/link/w/i/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://yun.139.com/link/w/i/{#shareCode#}",
          copyUrl:
            "https://yun.139.com/link/w/i/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("hecaiyun"),
        },
      ],
      aliyun: [
        {
          enable: NetDiskLocalData.enable("aliyun"),
          link_innerText: `aliyundrive.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "aliyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "aliyun"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `aliyundrive.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "aliyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "aliyun"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /aliyundrive\.com\/s\/([a-zA-Z0-9_\-]{8,14})/g,
          shareCodeNeedRemoveStr: /aliyundrive\.com\/s\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "aliyundrive.com/s/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://www.aliyundrive.com/s/{#shareCode#}",
          copyUrl:
            "https://www.aliyundrive.com/s/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("aliyun"),
        },
        {
          enable: NetDiskLocalData.enable("aliyun"),
          link_innerText: `aliyundrive.com/t/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "aliyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "aliyun"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `aliyundrive.com/t/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "aliyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "aliyun"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /aliyundrive\.com\/t\/([a-zA-Z0-9_\-]{8,14})/g,
          shareCodeNeedRemoveStr: /aliyundrive\.com\/t\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "aliyundrive.com/t/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://www.aliyundrive.com/t/{#shareCode#}",
          copyUrl:
            "https://www.aliyundrive.com/t/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("aliyun"),
        },
        {
          enable: NetDiskLocalData.enable("aliyun"),
          link_innerText: `alipan.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "aliyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "aliyun"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `alipan.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "aliyun"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "aliyun"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /alipan\.com\/s\/([a-zA-Z0-9_\-]{8,14})/g,
          shareCodeNeedRemoveStr: /alipan\.com\/s\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "alipan.com/s/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://www.alipan.com/s/{#shareCode#}",
          copyUrl:
            "https://www.alipan.com/s/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("aliyun"),
        },
      ],
      wenshushu: [
        {
          enable: NetDiskLocalData.enable("wenshushu"),
          link_innerText: `(wenshushu.cn/f/([a-zA-Z0-9_-]{8,14})|wenshushu.cn/k/([a-zA-Z0-9_-]{8,14}))([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "wenshushu"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "wenshushu"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `(wenshushu.cn/f/([a-zA-Z0-9_-]{8,14})|wenshushu.cn/k/([a-zA-Z0-9_-]{8,14}))([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "wenshushu"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "wenshushu"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode:
            /wenshushu.cn\/f\/([a-zA-Z0-9_-]{8,14})|wenshushu.cn\/k\/([a-zA-Z0-9_-]{8,14})/gi,
          shareCodeNeedRemoveStr: /wenshushu.cn\/f\/|wenshushu.cn\/k\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /[0-9a-zA-Z]{4}/gi,
          uiLinkShow: "www.wenshushu.cn/f/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://www.wenshushu.cn/f/{#shareCode#}",
          copyUrl:
            "https://www.wenshushu.cn/f/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("wenshushu"),
        },
        {
          enable: NetDiskLocalData.enable("wenshushu"),
          link_innerText: `(wss.ink/f/([a-zA-Z0-9_-]{8,14})|ws28.cn/f/([a-zA-Z0-9_-]{8,14})|wss1.cn/f/([a-zA-Z0-9_-]{8,14})|ws59.cn/f/([a-zA-Z0-9_-]{8,14}))([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "wenshushu"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "wenshushu"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `(wss.ink/f/([a-zA-Z0-9_-]{8,14})|ws28.cn/f/([a-zA-Z0-9_-]{8,14})|wss1.cn/f/([a-zA-Z0-9_-]{8,14})|ws59.cn/f/([a-zA-Z0-9_-]{8,14}))([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "wenshushu"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "wenshushu"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode:
            /wss.ink\/f\/([a-zA-Z0-9_-]{8,14})|ws28.cn\/f\/([a-zA-Z0-9_-]{8,14})|wss1.cn\/f\/([a-zA-Z0-9_-]{8,14})|ws59.cn\/f\/([a-zA-Z0-9_-]{8,14})/gi,
          shareCodeNeedRemoveStr:
            /wss.ink\/f\/|ws28.cn\/f\/|wss1.cn\/f\/|ws59.cn\/f\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /[0-9a-zA-Z]{4}/gi,
          uiLinkShow: "www.wenshushu.cn/f/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://www.wenshushu.cn/f/{#shareCode#}",
          copyUrl:
            "https://www.wenshushu.cn/f/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("wenshushu"),
        },
      ],
      nainiu: [
        {
          enable: NetDiskLocalData.enable("nainiu"),
          link_innerText: `cowtransfer.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "nainiu"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "nainiu"
          )}}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `cowtransfer.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "nainiu"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "nainiu"
          )}}[0-9a-zA-Z]{4,6}|)`,
          shareCode: /cowtransfer.com\/s\/([a-zA-Z0-9_\-]{8,14})/gi,
          shareCodeNeedRemoveStr: /cowtransfer\.com\/s\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4,6})/gi,
          uiLinkShow: "cowtransfer.com/s/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://cowtransfer.com/s/{#shareCode#}",
          copyUrl:
            "https://cowtransfer.com/s/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("nainiu"),
        },
      ],
      _123pan: [
        {
          enable: NetDiskLocalData.enable("_123pan"),
          link_innerText: `123pan.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_123pan"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "_123pan"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `123pan.com/s/([a-zA-Z0-9_-]{8,14})([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_123pan"
          )}}(密码|访问码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "_123pan"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /123pan.com\/s\/([a-zA-Z0-9_\-]{8,14})/gi,
          shareCodeNeedRemoveStr: /123pan.com\/s\//gi,
          checkAccessCode: /(密码|访问码|提取码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "123pan.com/s/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://123pan.com/s/{#shareCode#}",
          copyUrl: "https://123pan.com/s/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("_123pan"),
        },
      ],
      weiyun: [
        {
          enable: NetDiskLocalData.enable("weiyun"),
          link_innerText: `weiyun.com/[0-9a-zA-Z-_]{7,24}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "weiyun"
          )}(访问码|密码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "weiyun"
          )}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `weiyun.com/[0-9a-zA-Z-_]{7,24}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "weiyun"
          )}(访问码|密码|提取码)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "weiyun"
          )}[0-9a-zA-Z]{4,6}|)`,
          shareCode: /weiyun.com\/([0-9a-zA-Z\-_]{7,24})/gi,
          shareCodeNeedRemoveStr: /weiyun.com\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4,6})/gi,
          uiLinkShow: "share.weiyun.com/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://share.weiyun.com/{#shareCode#}",
          copyUrl:
            "https://share.weiyun.com/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("weiyun"),
        },
      ],
      xunlei: [
        {
          enable: NetDiskLocalData.enable("xunlei"),
          link_innerText: `xunlei.com/s/[0-9a-zA-Z-_]{8,30}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "xunlei"
          )}}(\\?pwd=|访问码|提取码|密码|)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "xunlei"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `xunlei.com\/s\/[0-9a-zA-Z\-_]{8,30}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "xunlei"
          )}}(\\?pwd=|访问码|提取码|密码|)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "xunlei"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /xunlei.com\/s\/([0-9a-zA-Z\-_]{8,30})/gi,
          shareCodeNeedRemoveStr: /xunlei.com\/s\//gi,
          checkAccessCode: /(\?pwd=|提取码|密码|访问码)[\s\S]+/g,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow:
            "pan.xunlei.com/s/{#shareCode#}?pwd={#accessCode#} 提取码: {#accessCode#}",
          blank: "https://pan.xunlei.com/s/{#shareCode#}?pwd={#accessCode#}",
          copyUrl: "https://pan.xunlei.com/s/{#shareCode#}?pwd={#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("xunlei"),
        },
      ],
      _115pan: [
        {
          enable: NetDiskLocalData.enable("_115pan"),
          link_innerText: `115.com/s/[0-9a-zA-Z-_]{8,24}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_115pan"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "_115pan"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `115.com\/s\/[0-9a-zA-Z-_]{8,24}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_115pan"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "_115pan"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /115.com\/s\/([0-9a-zA-Z\-_]{8,24})/gi,
          shareCodeNeedRemoveStr: /115.com\/s\//gi,
          checkAccessCode: /(提取码|密码|\?password=|访问码)[\s\S]+/gi,
          accessCode: /(\?password=|)([0-9a-zA-Z]{4})/i,
          uiLinkShow: "115.com/s/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://115.com/s/{#shareCode#}",
          copyUrl: "https://115.com/s/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("_115pan"),
        },
      ],
      chengtong: [
        /* d */
        {
          enable: NetDiskLocalData.enable("chengtong"),
          link_innerText: `(ct.ghpym.com|pan.jc-box.com|download.jamcz.com)/d/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `(ct.ghpym.com|pan.jc-box.com|download.jamcz.com)/d/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          shareCode:
            /(ct.ghpym.com|pan.jc-box.com|download.jamcz.com)\/d\/([0-9a-zA-Z\-_]{8,26})/gi,
          shareCodeNeedRemoveStr:
            /(ct.ghpym.com|pan.jc-box.com|download.jamcz.com)\/d\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4,6})/gi,
          paramMatch: /([a-zA-Z0-9\.]+)\/d\//i,
          uiLinkShow: "{#$1#}/d/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://{#$1#}/d/{#shareCode#}",
          copyUrl: "https://{#$1#}/d/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("chengtong"),
        },
        /* d */
        {
          enable: NetDiskLocalData.enable("chengtong"),
          link_innerText: `ctfile.com/d/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `ctfile.com/d/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          shareCode: /ctfile.com\/d\/([0-9a-zA-Z\-_]{8,26})/gi,
          shareCodeNeedRemoveStr: /ctfile.com\/d\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4,6})/gi,
          uiLinkShow: "url95.ctfile.com/d/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://url95.ctfile.com/d/{#shareCode#}",
          copyUrl:
            "https://url95.ctfile.com/d/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("chengtong"),
        },
        /* file */
        {
          enable: NetDiskLocalData.enable("chengtong"),
          link_innerText: `(2k.us|u062.com|545c.com|t00y.com)/file/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `(2k.us|u062.com|545c.com|t00y.com)/file/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          shareCode:
            /(2k.us|u062.com|545c.com|t00y.com)\/file\/([0-9a-zA-Z\-_]{8,26})/gi,
          shareCodeNeedRemoveStr:
            /(2k.us|u062.com|545c.com|t00y.com)\/file\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4,6})/gi,
          uiLinkShow: "u062.com/file/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://u062.com/file/{#shareCode#}",
          copyUrl: "https://u062.com/file/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("chengtong"),
        },
        /* f */
        {
          enable: NetDiskLocalData.enable("chengtong"),
          link_innerText: `(pan.jc-box.com|545c.com|down.jc-box.com|download.cx05.cc)/f/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `(pan.jc-box.com|545c.com|down.jc-box.com|download.cx05.cc)/f/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          shareCode:
            /(pan.jc-box.com|545c.com|down.jc-box.com|download.cx05.cc)\/f\/([0-9a-zA-Z\-_]{8,26})/gi,
          shareCodeNeedRemoveStr:
            /(pan.jc-box.com|545c.com|down.jc-box.com|download.cx05.cc)\/f\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4,6})/gi,
          paramMatch: /([0-9a-zA-Z\.+])\/f\//i,
          uiLinkShow: "{#$1#}/f/{#shareCode#} 提取码: {#accessCode#}",
          blank: "http://{#$1#}/f/{#shareCode#}",
          copyUrl: "http://{#$1#}/f/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("chengtong"),
        },
        /* f */
        {
          enable: NetDiskLocalData.enable("chengtong"),
          link_innerText: `(ctfile.com|089u.com)/f/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `(ctfile.com|089u.com)/f/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          shareCode: /(ctfile.com|089u.com)\/f\/([0-9a-zA-Z\-_]{8,26})/gi,
          shareCodeNeedRemoveStr: /(ctfile.com|089u.com)\/f\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4,6})/gi,
          uiLinkShow: "url95.ctfile.com/f/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://url95.ctfile.com/f/{#shareCode#}",
          copyUrl:
            "https://url95.ctfile.com/f/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("chengtong"),
        },
        /* dir */
        {
          enable: NetDiskLocalData.enable("chengtong"),
          link_innerText: `(089u.com|474b.com)/dir/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{4,6}|)`,
          link_innerHTML: `(089u.com|474b.com)/dir/[0-9a-zA-Z-_]{8,26}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "_chengtong"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "chengtong"
          )}}[0-9a-zA-Z]{6}|)`,
          shareCode: /(089u.com|474b.com)\/dir\/([0-9a-zA-Z\-_]{8,26})/gi,
          shareCodeNeedRemoveStr: /(089u.com|474b.com)\/dir\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{6})/gi,
          uiLinkShow: "089u.com/dir/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://089u.com/dir/{#shareCode#}",
          copyUrl: "https://089u.com/dir/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("chengtong"),
        },
      ],
      kuake: [
        {
          enable: NetDiskLocalData.enable("kuake"),
          link_innerText: `quark.cn/s/[0-9a-zA-Z-_]{8,24}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "kuake"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "kuake"
          )}}[0-9a-zA-Z]{4}|)`,
          link_innerHTML: `quark.cn/s/[0-9a-zA-Z-_]{8,24}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "kuake"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "kuake"
          )}}[0-9a-zA-Z]{4}|)`,
          shareCode: /quark.cn\/s\/([0-9a-zA-Z\-_]{8,24})/gi,
          shareCodeNeedRemoveStr: /quark.cn\/s\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "quark.cn/s/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://pan.quark.cn/s/{#shareCode#}",
          copyUrl: "https://pan.quark.cn/s/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("kuake"),
        },
      ],
      magnet: [
        {
          enable: NetDiskLocalData.enable("magnet"),
          link_innerText: `magnet:\\?xt=urn:btih:[0-9a-fA-F]{32,40}`,
          link_innerHTML: `magnet:\\?xt=urn:btih:[0-9a-fA-F]{32,40}`,
          shareCode: /magnet:\?xt=urn:btih:([0-9a-fA-F]{32,40})/gi,
          shareCodeNeedRemoveStr: /magnet:\?xt=urn:btih:/gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4})/gi,
          uiLinkShow: "magnet:?xt=urn:btih:{#shareCode#}",
          blank: "magnet:?xt=urn:btih:{#shareCode#}",
          copyUrl: "magnet:?xt=urn:btih:{#shareCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("magnet"),
        },
      ],
      jianguoyun: [
        {
          enable: NetDiskLocalData.enable("jianguoyun"),
          link_innerText: `jianguoyun.com/p/[0-9a-zA-Z-_]{16,24}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "jianguoyun"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "jianguoyun"
          )}}[0-9a-zA-Z]+|)`,
          link_innerHTML: `jianguoyun.com/p/[0-9a-zA-Z-_]{16,24}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "jianguoyun"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "jianguoyun"
          )}}[0-9a-zA-Z]+|)`,
          shareCode: /jianguoyun.com\/p\/([0-9a-zA-Z\-_]{16,24})/gi,
          shareCodeNeedRemoveStr: /jianguoyun.com\/p\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{3,6})/gi,
          uiLinkShow: "jianguoyun.com/p/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://www.jianguoyun.com/p/{#shareCode#}",
          copyUrl:
            "https://www.jianguoyun.com/p/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("jianguoyun"),
        },
      ],
      onedrive: [
        {
          enable: NetDiskLocalData.enable("onedrive"),
          link_innerText: `[0-9a-zA-Z-_]+.sharepoint.com/[0-9a-zA-Z-_:]+/[0-9a-zA-Z-_:]+/personal/[0-9a-zA-Z-_]+/[0-9a-zA-Z-_]+([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "onedrive"
          )}}(访问码|密码|提取码|\\?password=\\?e=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "onedrive"
          )}}[0-9a-zA-Z]+|)`,
          link_innerHTML: `[0-9a-zA-Z-_]+.sharepoint.com/[0-9a-zA-Z-_:]+/[0-9a-zA-Z-_:]+/personal/[0-9a-zA-Z-_]+/[0-9a-zA-Z-_]+([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "onedrive"
          )}}(访问码|密码|提取码|\\?password=\\?e=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "onedrive"
          )}}[0-9a-zA-Z]+|)`,
          shareCode:
            /[0-9a-zA-Z-_]+\/[0-9a-zA-Z-_:]+\/[0-9a-zA-Z-_:]+\/personal\/[0-9a-zA-Z-_]+\/([0-9a-zA-Z\-_]+)/gi,
          shareCodeNeedRemoveStr:
            /[0-9a-zA-Z-_]+\/[0-9a-zA-Z-_:]+\/[0-9a-zA-Z-_:]+\/personal\/[0-9a-zA-Z-_]+\//gi,
          checkAccessCode: /(提取码|密码|访问码|\?password=|\?e=)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]{4,8})/gi,
          paramMatch:
            /([0-9a-zA-Z-_]+).sharepoint.com\/([0-9a-zA-Z-_:]+)\/([0-9a-zA-Z-_:]+)\/personal\/([0-9a-zA-Z-_]+)\/([0-9a-zA-Z-_]+)/i,
          uiLinkShow:
            "{#$1#}.sharepoint.com/{#$2#}/{#$3#}/personal/{#$4#}/{#shareCode#} 提取码: {#accessCode#}",
          blank:
            "https://{#$1#}.sharepoint.com/{#$2#}/{#$3#}/personal/{#$4#}/{#shareCode#}?e={#accessCode#}",
          copyUrl:
            "https://{#$1#}.sharepoint.com/{#$2#}/{#$3#}/personal/{#$4#}/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("onedrive"),
        },
      ],
      uc: [
        {
          enable: NetDiskLocalData.enable("uc"),
          link_innerText: `(drive|fast).uc.cn/s/[0-9a-zA-Z]{8,24}([\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
            "uc"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
            "uc"
          )}}[0-9a-zA-Z]+|)`,
          link_innerHTML: `(drive|fast).uc.cn/s/[0-9a-zA-Z]{8,24}([\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
            "uc"
          )}}(访问码|密码|提取码|\\?password=)[\\s\\S]{0,${NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
            "uc"
          )}}[0-9a-zA-Z]+|)`,
          shareCode: /(drive|fast).uc.cn\/s\/([0-9a-zA-Z]{8,24})/gi,
          shareCodeNeedRemoveStr: /(drive|fast).uc.cn\/s\//gi,
          checkAccessCode: /(提取码|密码|访问码)[\s\S]+/gi,
          accessCode: /([0-9a-zA-Z]+)/gi,
          uiLinkShow: "drive.uc.cn/s/{#shareCode#} 提取码: {#accessCode#}",
          blank: "https://drive.uc.cn/s/{#shareCode#}",
          copyUrl: "https://drive.uc.cn/s/{#shareCode#}\n密码:{#accessCode#}",
          checkLinkValidity: NetDiskLocalData.checkLinkValidity("uc"),
        },
      ],
    },
    /**
     * 初始化
     */
    init() {
      if (!this.isInit) {
        this.isInit = true;
        this.matchLink = new Set();
        this.initLinkDict();
      }
    },
    /**
     * 初始化字典
     */
    initLinkDict() {
      NetDisk.linkDict = new utils.Dictionary();
      NetDisk.tempLinkDict = new utils.Dictionary();
      Object.keys(NetDisk.regular).forEach((netDiskName) => {
        NetDisk.linkDict.set(netDiskName, new utils.Dictionary());
        NetDisk.tempLinkDict.set(netDiskName, new utils.Dictionary());
      });
    },
    /**
     * 删除某些需要忽略的text或html,如:设置、直链弹窗
     * @param {string} text 需要进行处理的字符串
     * @param {boolean} isHTML 是否是html属性
     * @returns {string}
     */
    ignoreStrRemove(text, isHTML = false) {
      let ignoreNodeList = [];
      ignoreNodeList.forEach((ignoreElement) => {
        if (ignoreElement == void 0) {
          return;
        }
        if (isHTML) {
          if (ignoreElement.innerHTML != void 0) {
            text = text.replaceAll(ignoreElement.innerHTML, "");
          }
        } else {
          let text = ignoreElement.innerText || ignoreElement.textContent;
          if (text != void 0) {
            text = text.replaceAll(text, "");
          }
        }
      });
      return text;
    },
    /**
     * 处理链接,将匹配到的链接转为参数和密码存入字典中
     * @param {string} netDiskName 网盘名称
     * @param {number} netDiskIndex 网盘名称的索引下标
     * @param {string} matchText 正在进行匹配的文本
     * @returns 返回值可能为空
     */
    handleLink(netDiskName, netDiskIndex, matchText) {
      let shareCode = this.handleShareCode(
        netDiskName,
        netDiskIndex,
        matchText
      );
      if (utils.isNull(shareCode)) {
        return;
      }
      let accessCode = this.handleAccessCode(
        netDiskName,
        netDiskIndex,
        matchText
      );
      accessCode = this.handleAccessCodeByUserRule(
        netDiskName,
        accessCode,
        matchText
      );
      return {
        shareCode: shareCode,
        accessCode: accessCode,
      };
    },
    /**
     * 对传入的url进行处理,返回shareCode
     * @param {string} netDiskName 网盘名称
     * @param {number} netDiskIndex 网盘名称索引下标
     * @param {string} matchText 正在进行匹配的文本
     */
    handleShareCode(netDiskName, netDiskIndex, matchText) {
      /* 当前执行的规则 */
      let netDiskMatchRegular = NetDisk.regular[netDiskName][netDiskIndex];
      let shareCodeMatch = matchText
        .match(netDiskMatchRegular.shareCode)
        ?.filter((item) => utils.isNotNull(item));
      if (utils.isNull(shareCodeMatch)) {
        log.error([
          `匹配shareCode为空`,
          {
            匹配的文本: matchText,
            规则: netDiskMatchRegular,
            正在使用的规则: netDiskMatchRegular.shareCode,
            网盘名称: netDiskName,
            网盘名称索引下标: netDiskIndex,
          },
        ]);
        return;
      }
      /* 匹配到的网盘链接,取第一个值就行 */
      let shareCode = shareCodeMatch[0];
      if (netDiskMatchRegular.shareCodeNeedRemoveStr) {
        /* 删除ShareCode前面不需要的字符串 */
        shareCode = shareCode.replace(
          netDiskMatchRegular.shareCodeNeedRemoveStr,
          ""
        );
      }
      let shareCodeNotMatch = netDiskMatchRegular.shareCodeNotMatch;
      if (shareCodeNotMatch != void 0 && shareCode.match(shareCodeNotMatch)) {
        log.error(`不可能的shareCode => ${shareCode}`);
        return;
      }
      for (const shareCodeNotMatchRegexp of NetDisk.shareCodeNotMatchRegexpList) {
        if (shareCode.match(shareCodeNotMatchRegexp)) {
          log.error(`不可能的shareCode => ${shareCode}`);
          return;
        }
      }
      /* %E7%BD%91%E7%9B%98 => 网盘 */
      shareCode = decodeURI(shareCode);
      if (
        GM_getValue("excludeIdenticalSharedCodes") &&
        utils.isSameChars(
          shareCode,
          GM_getValue(
            "excludeIdenticalSharedCodesCoefficient",
            NetDiskData.defaultExcludeIdenticalSharedCodesCoefficient
          )
        )
      ) {
        /* 排除掉由相同字符组成的分享码 */
        return;
      }
      /* 排除掉以http|https结尾的分享码 */
      if (shareCode.endsWith("http") || shareCode.endsWith("https")) {
        return;
      }
      return shareCode;
    },
    /**
     * 对传入的url进行处理,返回accessCode
     * @param {string} netDiskName 网盘名称
     * @param {number} netDiskIndex 网盘名称索引下标
     * @param {string} matchText 正在进行匹配的文本
     * @returns {string} "xxxx" || ""
     */
    handleAccessCode(netDiskName, netDiskIndex, matchText) {
      /* 当前执行正则匹配的规则 */
      let netDiskMatchRegular = this.regular[netDiskName][netDiskIndex];
      let accessCode = "";
      if (!netDiskMatchRegular.checkAccessCode) {
        /* 不存在匹配提取码的正则 */
        return "";
      }
      let accessCodeMatch = matchText.match(
        netDiskMatchRegular.checkAccessCode
      );
      if (accessCodeMatch) {
        /* 匹配出带提取码的字符串 */
        let accessCodeMatchValue = accessCodeMatch[accessCodeMatch.length - 1];
        /* 进去提取码匹配,且过滤掉null或undefined或空字符串 */
        let accessCodeMatchArray = accessCodeMatchValue
          .match(netDiskMatchRegular.accessCode)
          ?.filter((item) => utils.isNotNull(item));
        if (utils.isNull(accessCodeMatchArray)) {
          return "";
        }
        if (accessCodeMatchArray.length) {
          /* 取第一个值 */
          /**
           * 例如,匹配到的字符串是密码:oanm   大于150m
           * 如果是最后一个,那么会匹配到150m
           */
          accessCode = accessCodeMatchArray[0];
          if (accessCode.startsWith("http")) {
            /* 排除不可能的accessCode */
            accessCode = "";
          }
        }
      }
      if (utils.isNotNull(accessCode)) {
        for (const accessCodeNotMatchRegexp of NetDisk.accessCodeNotMatchRegexpList) {
          if (accessCode.match(accessCodeNotMatchRegexp)) {
            accessCode = "";
            break;
          }
        }
        if (
          netDiskMatchRegular.acceesCodeNotMatch &&
          accessCode.match(netDiskMatchRegular.acceesCodeNotMatch)
        ) {
          accessCode = "";
        }
      }
      return accessCode;
    },
    /**
     * 对accessCode二次处理,使用自定义的访问码规则
     * @param {string} netDiskName 网盘名称
     * @param {string} accessCode 访问码
     * @param {string} matchText 匹配到的文本
     * @returns {string}
     */
    handleAccessCodeByUserRule(netDiskName, accessCode, matchText) {
      let regularList = NetDiskUI.accessCodeRule.getValue();
      let result = accessCode;
      let currentUrl = window.location.href;
      /* 先遍历本地的自定义的访问码规则 */
      for (
        let regularIndex = 0;
        regularIndex < regularList.length;
        regularIndex++
      ) {
        let rule = regularList[regularIndex];
        let urlRegexp = new RegExp(rule.urlRegexp, "i");
        /* 如果网址匹配成功则进行下一步 */
        if (!currentUrl.match(urlRegexp)) {
          continue;
        }
        /* 循环遍历自定义的访问码规则的网盘信息 */
        for (let index = 0; index < rule.netdisk.length; index++) {
          let netDiskRegular = rule.netdisk[index];
          /* 自定义规则的value(也就是网盘的字母名)和参数netDiskName相同,则直接返回设定的值 */
          if (netDiskRegular.value === netDiskName) {
            log.success(`使用自定义规则中的提取码 ${netDiskName} ${result}`);
            return rule.accessCode;
          }
        }
      }
      /* 不存在 */
      return result;
    },
    /**
     * 获取在弹窗中显示出的链接
     * @param {string} netDiskName 网盘名称,指NetDisk.regular的内部键名
     * @param {number} netDiskIndex 网盘名称索引下标
     * @param {string} shareCode 分享码
     * @param {string} accessCode 访问码
     * @param {?string} matchText 匹配到的文本
     * @returns {string}
     */
    handleLinkShow(
      netDiskName,
      netDiskIndex,
      shareCode,
      accessCode,
      matchText
    ) {
      let netDiskMatchRegular = NetDisk.regular[netDiskName][netDiskIndex];
      if (netDiskMatchRegular == void 0) {
        Qmsg.error("BUG: 获取uiLink规则失败");
        log.error([
          "BUG: 分析参数",
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode,
        ]);
        throw "获取uiLink规则失败";
      }
      let uiLink = this.replaceParam(netDiskMatchRegular["uiLinkShow"], {
        shareCode: shareCode,
      });
      if (accessCode && accessCode != "") {
        uiLink = this.replaceParam(uiLink, {
          accessCode: accessCode,
        });
      } else {
        uiLink = uiLink.replace(NetDisk.noAccessCodeRegExp, "");
      }
      if (netDiskMatchRegular.paramMatch) {
        /**
         * 当前字典
         */
        let currentDict = NetDisk.linkDict.get(netDiskName).get(shareCode);
        matchText = matchText ?? currentDict?.matchText;
        if (utils.isNotNull(matchText)) {
          let paramMatchArray = matchText.match(netDiskMatchRegular.paramMatch);
          let replaceParamData = {};
          for (let index = 0; index < paramMatchArray.length; index++) {
            replaceParamData[`$${index}`] = paramMatchArray[index];
          }
          uiLink = this.replaceParam(uiLink, replaceParamData);
        }
      }
      return uiLink;
    },
    /**
     * 参数替换,区分大小写
     *
     * 例如
     * + {#shareCode#} => xxxx
     * + {#accessCode#} => xxxx
     * + {#$1#} => xxxx
     * @param {string} text
     * @param {{}} [data={}]
     */
    replaceParam(text, data = {}) {
      if (typeof text !== "string") {
        return text;
      }
      Object.keys(data).forEach((key) => {
        let replacedText = data[key];
        if (utils.isNotNull(replacedText)) {
          try {
            text = text.replaceAll(
              `{#encodeURI-${key}#}`,
              encodeURI(replacedText)
            );
          } catch (error) {
            log.error(["encodeURI-替换的文本失败", [replacedText]]);
          }

          try {
            text = text.replaceAll(
              `{#encodeURIComponent-${key}#}`,
              encodeURIComponent(replacedText)
            );
          } catch (error) {
            log.error(["encodeURIComponent-替换的文本失败", [replacedText]]);
          }
          try {
            text = text.replaceAll(
              `{#decodeURI-${key}#}`,
              decodeURI(replacedText)
            );
          } catch (error) {
            log.error(["decodeURI-替换的文本失败", [replacedText]]);
          }
          try {
            text = text.replaceAll(
              `{#decodeURIComponent-${key}#}`,
              decodeURIComponent(replacedText)
            );
          } catch (error) {
            log.error(["encodeURIComponent-替换的文本失败", [replacedText]]);
          }
          text = text.replaceAll(`{#${key}#}`, replacedText);
        }
      });
      return text;
    },
    /**
     * 获取剪贴板文本
     * @returns {Promise<string>}
     */
    async getClipboardText() {
      /** 读取剪贴板 */
      function readClipboardText(resolve) {
        navigator.clipboard
          .readText()
          .then((clipboardText) => {
            resolve(clipboardText);
          })
          .catch(
            /** @param {TypeError} error */
            (error) => {
              log.error(["读取剪贴板内容失败👉", error]);
              resolve("");
            }
          );
      }
      /** 申请读取剪贴板的权限 */
      function requestPermissionsWithClipboard(resolve) {
        navigator.permissions
          .query({
            name: "clipboard-read",
          })
          .then((permissionStatus) => {
            readClipboardText(resolve);
          })
          .catch(
            /** @param {TypeError} error */
            (error) => {
              log.error([
                "申请剪贴板权限失败,尝试直接读取👉",
                error.message ?? error.name ?? error.stack,
              ]);
              /* 该权限申请Api可能在该环境下不生效,尝试直接读取剪贴板 */
              readClipboardText(resolve);
            }
          );
      }
      function checkClipboardApi() {
        if (typeof navigator?.clipboard?.readText !== "function") {
          return false;
        }
        if (typeof navigator?.permissions?.query !== "function") {
          return false;
        }
        return true;
      }
      return new Promise((resolve) => {
        if (!checkClipboardApi()) {
          resolve("");
          return;
        }
        if (document.hasFocus()) {
          requestPermissionsWithClipboard(resolve);
        } else {
          window.addEventListener(
            "focus",
            () => {
              requestPermissionsWithClipboard(resolve);
            },
            {
              once: true,
            }
          );
        }
      });
    },
    /**
     * 获取已匹配到的链接的存储的对象
     * @param {string} accessCode 访问码
     * @param {number} [netDiskIndex=0] 下标,默认0
     * @param {boolean} [isForceAccessCode=false] 是否锁定访问码不允许修改,默认false
     * @param {string} matchText 匹配到的文本
     * @returns {NetDiskDictData}
     */
    getLinkDickObj(
      accessCode,
      netDiskIndex = 0,
      isForceAccessCode = false,
      matchText
    ) {
      return {
        accessCode: accessCode,
        netDiskIndex: netDiskIndex,
        isForceAccessCode: isForceAccessCode,
        matchText: matchText,
      };
    },
  };

  const NetDiskDebug = {
    /**
     * 对传入的url进行处理,返回shareCode
     * @param {string} matchText 正在进行匹配的文本
     * @param {NetDiskRegularOption} regular 当前执行的规则
     * @param {(logData: NetDiskDebugLogData)=>void} logCallBack 日志回调
     */
    handleShareCode(matchText, regular, logCallBack) {
      let shareCodeMatch = matchText
        .match(regular.shareCode)
        ?.filter((item) => utils.isNotNull(item));
      logCallBack({
        status: true,
        msg: [
          `正则: shareCode`,
          "作用: 获取shareCode",
          "结果: ",
          shareCodeMatch,
        ],
      });
      if (utils.isNull(shareCodeMatch)) {
        logCallBack({
          status: false,
          msg: `匹配shareCode为空`,
        });
        return;
      }
      /* 匹配到的网盘链接,取第一个值就行 */
      let shareCode = shareCodeMatch[0];
      logCallBack({
        status: true,
        msg: [`取第一个值: ` + shareCode],
      });
      if (regular.shareCodeNeedRemoveStr) {
        /* 删除ShareCode前面不需要的字符串 */
        shareCode = shareCode.replace(regular.shareCodeNeedRemoveStr, "");
        logCallBack({
          status: true,
          msg: [
            `正则: shareCodeNeedRemoveStr`,
            "作用: 删除ShareCode前面不需要的字符串",
            `结果: ${shareCode}`,
          ],
        });
      }
      let shareCodeNotMatch = regular.shareCodeNotMatch;
      if (shareCodeNotMatch != void 0 && shareCode.match(shareCodeNotMatch)) {
        log.error(`不可能的shareCode => ${shareCode}`);
        logCallBack({
          status: false,
          msg: [
            `正则: shareCodeNotMatch`,
            "作用: 用于判断提取到的shareCode是否是错误的shareCode",
            `结果: true 该shareCode不是正确的`,
          ],
        });
        return;
      }
      for (const shareCodeNotMatchRegexp of NetDisk.shareCodeNotMatchRegexpList) {
        if (shareCode.match(shareCodeNotMatchRegexp)) {
          log.error(`不可能的shareCode => ${shareCode}`);
          logCallBack({
            status: false,
            msg: [
              `正则: 内置的shareCodeNotMatchRegexpList`,
              "作用: 使用该正则判断提取到的shareCode是否正确",
              `结果: true 该shareCode不是正确的`,
            ],
          });
          return;
        }
      }
      /* %E7%BD%91%E7%9B%98 => 网盘 */
      shareCode = decodeURI(shareCode);
      logCallBack({
        status: true,
        msg: ["对shareCode进行解码: " + shareCode],
      });
      if (
        GM_getValue("excludeIdenticalSharedCodes") &&
        utils.isSameChars(
          shareCode,
          GM_getValue(
            "excludeIdenticalSharedCodesCoefficient",
            NetDiskData.defaultExcludeIdenticalSharedCodesCoefficient
          )
        )
      ) {
        /* 排除掉由相同字符组成的分享码 */
        logCallBack({
          status: false,
          msg: ["已开启【排除分享码】且该分享码命中该规则"],
        });
        return;
      }
      /* 排除掉以http|https结尾的分享码 */
      if (shareCode.endsWith("http") || shareCode.endsWith("https")) {
        logCallBack({
          status: false,
          msg: ["该分享码以http|https结尾"],
        });
        return;
      }
      logCallBack({
        status: true,
        msg: "处理完毕的shareCode: " + shareCode,
      });
      return shareCode;
    },
    /**
     * 对传入的url进行处理,返回accessCode
     * @param {string} matchText 正在进行匹配的文本
     * @param {NetDiskRegularOption} regular 当前执行的规则
     * @param {(logData: NetDiskDebugLogData)=>void} logCallBack 日志回调
     */
    handleAccessCode(matchText, regular, logCallBack) {
      let accessCode = "";
      if (!regular.checkAccessCode) {
        /* 不存在匹配提取码的正则 */
        logCallBack({
          status: true,
          msg: "因未配置规则checkAccessCode,默认accessCode的值为空",
        });
        return "";
      }
      let accessCodeMatch = matchText.match(regular.checkAccessCode);
      logCallBack({
        status: true,
        msg: [
          `正则: checkAccessCode`,
          "作用: 用来判断link_innerText或者link_innerHTML匹配到的字符串中是否存在密码",
          `结果: `,
          accessCodeMatch,
        ],
      });
      if (accessCodeMatch) {
        /* 匹配出带提取码的字符串 */
        let accessCodeMatchValue = accessCodeMatch[accessCodeMatch.length - 1];
        logCallBack({
          status: true,
          msg: "取最后一个值: " + accessCodeMatchValue,
        });
        /* 进去提取码匹配,且过滤掉null或undefined或空字符串 */
        let accessCodeMatchArray = accessCodeMatchValue
          .match(regular.accessCode)
          ?.filter((item) => utils.isNotNull(item));
        logCallBack({
          status: true,
          msg: [
            `正则: accessCode`,
            "作用: 用来提取link_innerText或者link_innerHTML匹配到的字符串中的密码",
            `结果: `,
            accessCodeMatchArray,
          ],
        });
        if (utils.isNull(accessCodeMatchArray)) {
          logCallBack({
            status: true,
            msg: "因↑匹配到的结果为空,默认accessCode的值为空",
          });
          return "";
        }
        if (accessCodeMatchArray.length) {
          /* 取第一个值 */
          /**
           * 例如,匹配到的字符串是密码:oanm   大于150m
           * 如果是最后一个,那么会匹配到150m
           */
          accessCode = accessCodeMatchArray[0];
          logCallBack({
            status: true,
            msg: "取第一个值: " + accessCode,
          });
          if (accessCode.startsWith("http")) {
            /* 排除不可能的accessCode */
            logCallBack({
              status: true,
              msg: "排除不可能的accessCode,重置accessCode的值为空",
            });
            accessCode = "";
          }
        }
      }
      if (utils.isNotNull(accessCode)) {
        for (const accessCodeNotMatchRegexp of NetDisk.accessCodeNotMatchRegexpList) {
          if (accessCode.match(accessCodeNotMatchRegexp)) {
            accessCode = "";
            logCallBack({
              status: true,
              msg: [
                `正则: 内置的accessCodeNotMatchRegexpList`,
                "作用: 使用该正则判断提取到的accessCode是否正确",
                `结果: true 重置accessCode为空`,
              ],
            });
            break;
          }
        }
        if (
          regular.acceesCodeNotMatch &&
          accessCode.match(regular.acceesCodeNotMatch)
        ) {
          accessCode = "";
          logCallBack({
            status: true,
            msg: [
              `正则: acceesCodeNotMatch`,
              "作用: 用于判断提取到的accessCode是否是错误的accessCode",
              `结果: true 重置accessCode为空`,
            ],
          });
        }
      }

      logCallBack({
        status: true,
        msg: "处理完毕的accessCode: " + accessCode,
      });
      return accessCode;
    },
    /**
     * 获取在弹窗中显示出的链接
     * @param {string} matchText 匹配到的文本
     * @param {NetDiskRegularOption} regular 当前执行的规则
     * @param {string} shareCode 分享码
     * @param {string} accessCode 访问码
     * @param {(logData: NetDiskDebugLogData)=>void} logCallBack 日志回调
     */
    handleLinkShow(matchText, regular, shareCode, accessCode, logCallBack) {
      let uiLink = NetDisk.replaceParam(regular["uiLinkShow"], {
        shareCode: shareCode,
      });
      logCallBack({
        status: true,
        msg: [
          `正则: uiLinkShow`,
          "作用: 用于显示在弹窗中的字符串",
          "备注: 对shareCode进行参数替换",
          `结果: ${uiLink}`,
        ],
      });
      if (accessCode && accessCode != "") {
        uiLink = NetDisk.replaceParam(uiLink, {
          accessCode: accessCode,
        });
        logCallBack({
          status: true,
          msg: [
            `正则: uiLinkShow`,
            "作用: 用于显示在弹窗中的字符串",
            "备注: 对accessCode进行参数替换",
            `结果: ${uiLink}`,
          ],
        });
      } else {
        uiLink = uiLink.replace(NetDisk.noAccessCodeRegExp, "");
        logCallBack({
          status: true,
          msg: [
            `正则: 内置的noAccessCodeRegExp`,
            "作用: 因accessCode为空,使用该正则去除不需要的字符串",
            `结果: ${uiLink}`,
          ],
        });
      }
      if (regular.paramMatch) {
        if (utils.isNotNull(matchText)) {
          let paramMatchArray = matchText.match(regular.paramMatch);
          let replaceParamData = {};
          for (let index = 0; index < paramMatchArray.length; index++) {
            replaceParamData[`$${index}`] = paramMatchArray[index];
          }
          uiLink = NetDisk.replaceParam(uiLink, replaceParamData);
          logCallBack({
            status: true,
            msg: [
              `正则: paramMatch`,
              `作用: 用于对matchText进行提取需要的关键内容,替换关键字:{#$1#}、{#$2#}...`,
              `参数: ` + JSON.stringify(replaceParamData, void 0, 4),
              `结果: ${uiLink}`,
            ],
          });
        }
      }

      logCallBack({
        status: true,
        msg: "处理完毕的uiLink: " + uiLink,
      });
      return uiLink;
    },
    /**
     * 获取新标签页打开的URL
     * @param {string} matchText 匹配到的文本
     * @param {NetDiskRegularOption} regular 当前执行的规则
     * @param {string} shareCode 分享码
     * @param {string} accessCode 访问码
     * @param {(logData: NetDiskDebugLogData)=>void} logCallBack 日志回调
     */
    handleBlank(matchText, regular, shareCode, accessCode, logCallBack) {
      let blankUrl = NetDisk.replaceParam(regular["blank"], {
        shareCode: shareCode,
      });
      logCallBack({
        status: true,
        msg: [
          `正则: blank`,
          "作用: 用于点击跳转的链接",
          "备注: 对shareCode进行参数替换",
          `结果: ${blankUrl}`,
        ],
      });
      if (accessCode && accessCode != "") {
        blankUrl = NetDisk.replaceParam(blankUrl, {
          accessCode: accessCode,
        });
        logCallBack({
          status: true,
          msg: [
            `正则: blank`,
            "作用: 用于点击跳转的链接",
            "备注: 对accessCode进行参数替换",
            `结果: ${blankUrl}`,
          ],
        });
      } else {
        blankUrl = blankUrl.replace(NetDisk.noAccessCodeRegExp, "");
        logCallBack({
          status: true,
          msg: [
            `正则: 内置的noAccessCodeRegExp`,
            "作用: 因accessCode为空,使用该正则去除不需要的字符串",
            `结果: ${blankUrl}`,
          ],
        });
      }
      if (regular.paramMatch) {
        if (utils.isNotNull(matchText)) {
          let paramMatchArray = matchText.match(regular.paramMatch);
          let replaceParamData = {};
          for (let index = 0; index < paramMatchArray.length; index++) {
            replaceParamData[`$${index}`] = paramMatchArray[index];
          }
          blankUrl = NetDisk.replaceParam(blankUrl, replaceParamData);
          logCallBack({
            status: true,
            msg: [
              `正则: paramMatch`,
              `作用: 用于对matchText进行提取需要的关键内容,替换关键字:{#$1#}、{#$2#}...`,
              `参数: ` + JSON.stringify(replaceParamData, void 0, 4),
              `结果: ${blankUrl}`,
            ],
          });
        }
      }

      logCallBack({
        status: true,
        msg: "处理完毕的blank: " + blankUrl,
      });
      return blankUrl;
    },
    /**
     * 获取复制到剪贴板的字符串
     * @param {string} matchText 匹配到的文本
     * @param {NetDiskRegularOption} regular 当前执行的规则
     * @param {string} shareCode 分享码
     * @param {string} accessCode 访问码
     * @param {(logData: NetDiskDebugLogData)=>void} logCallBack 日志回调
     */
    handleCopyUrl(matchText, regular, shareCode, accessCode, logCallBack) {
      let copyUrl = NetDisk.replaceParam(regular["copyUrl"], {
        shareCode: shareCode,
      });
      logCallBack({
        status: true,
        msg: [
          `正则: copyUrl`,
          "作用: 用于复制到剪贴板的链接",
          "备注: 对shareCode进行参数替换",
          `结果: ${copyUrl}`,
        ],
      });
      if (accessCode && accessCode != "") {
        copyUrl = NetDisk.replaceParam(copyUrl, {
          accessCode: accessCode,
        });
        logCallBack({
          status: true,
          msg: [
            `正则: copyUrl`,
            "作用: 用于复制到剪贴板的链接",
            "备注: 对accessCode进行参数替换",
            `结果: ${copyUrl}`,
          ],
        });
      } else {
        copyUrl = copyUrl.replace(NetDisk.noAccessCodeRegExp, "");
        logCallBack({
          status: true,
          msg: [
            `正则: 内置的noAccessCodeRegExp`,
            "作用: 因accessCode为空,使用该正则去除不需要的字符串",
            `结果: ${copyUrl}`,
          ],
        });
      }
      if (regular.paramMatch) {
        if (utils.isNotNull(matchText)) {
          let paramMatchArray = matchText.match(regular.paramMatch);
          let replaceParamData = {};
          for (let index = 0; index < paramMatchArray.length; index++) {
            replaceParamData[`$${index}`] = paramMatchArray[index];
          }
          copyUrl = NetDisk.replaceParam(copyUrl, replaceParamData);
          logCallBack({
            status: true,
            msg: [
              `正则: paramMatch`,
              `作用: 用于对matchText进行提取需要的关键内容,替换关键字:{#$1#}、{#$2#}...`,
              `参数: ` + JSON.stringify(replaceParamData, void 0, 4),
              `结果: ${copyUrl}`,
            ],
          });
        }
      }

      logCallBack({
        status: true,
        msg: "处理完毕的copyUrl: " + copyUrl,
      });
      return copyUrl;
    },
  };

  /** 网盘-校验链接有效性 */
  const NetDiskCheckLinkValidity = {
    $data: {
      isSubscribing: false,
      /**
       * @type {NetDiskCheckLinkValidityOption[]}
       */
      subscribe: [],
    },
    /**
     * 网盘检查的状态码
     */
    status: {
      loading: {
        code: 1,
        msg: "验证中",
        setView(ele, checkInfo) {
          NetDiskCheckLinkValidity.setViewCheckValid(ele, "loading");
          ele.innerHTML = pops.config.iconSVG.loading;
        },
      },
      success: {
        code: 200,
        msg: "该链接有效",
        setView(ele, checkInfo) {
          NetDiskCheckLinkValidity.setViewCheckValid(ele, "success");
          ele.innerHTML = `
          <svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg">
            <path
              fill="currentColor"
              d="M874.119618 149.859922A510.816461 510.816461 0 0 0 511.997 0.00208a509.910462 509.910462 0 0 0-362.119618 149.857842c-199.817789 199.679789-199.817789 524.581447 0 724.260236a509.969462 509.969462 0 0 0 362.119618 149.857842A508.872463 508.872463 0 0 0 874.119618 874.120158c199.836789-199.679789 199.836789-524.581447 0-724.260236zM814.94268 378.210681L470.999043 744.132295a15.359984 15.359984 0 0 1-5.887994 4.095996c-1.751998 1.180999-2.913997 2.362998-5.276994 2.913997a34.499964 34.499964 0 0 1-13.469986 2.914997 45.547952 45.547952 0 0 1-12.897986-2.303998l-4.095996-2.363997a45.291952 45.291952 0 0 1-7.009992-4.095996l-196.902793-193.789796a34.126964 34.126964 0 0 1-10.555989-25.186973c0-9.37399 3.583996-18.74698 9.98399-25.186974a36.429962 36.429962 0 0 1 50.372947 0l169.98382 167.423824L763.389735 330.220732a37.059961 37.059961 0 0 1 50.371947-1.732998 33.647965 33.647965 0 0 1 11.165988 25.186973 35.544963 35.544963 0 0 1-9.98399 24.575974v-0.04z m0 0"></path>
          </svg>
          `;
          NetDiskCheckLinkValidity.setViewAgainCheckClickEvent(ele, checkInfo);
        },
      },
      /**
       * @type {NetDiskCheckLinkValidityStatus}
       */
      error: {
        code: -404,
        msg: "网络异常",
        setView(ele, checkInfo) {
          NetDiskCheckLinkValidity.setViewCheckValid(ele, "error");
          ele.innerHTML = `
          <svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg">
            <path
              fill="currentColor"
              d="M511.808 692.224c-18.048 0-35.136 3.968-50.432 11.392-25.472 12.416-46.528 33.92-57.792 60.032-5.632 14.144-9.024 29.504-9.024 45.952 0 65.152 52.672 117.824 117.248 117.824 65.28 0 117.952-52.672 117.952-117.824 0-64.64-52.672-117.376-117.952-117.376z m0 178.496c-33.408 0-60.608-27.712-60.608-61.12 0-33.472 27.2-60.672 60.608-60.672s61.248 27.2 61.248 60.672c0 33.472-27.776 61.12-61.248 61.12zM286.784 661.632c3.968 3.392 8.512 5.632 12.992 5.632L438.08 523.328c-60.032 14.72-114.432 49.344-155.328 98.624-9.536 11.84-7.872 30.08 4.032 39.68zM622.912 534.656l-43.008 45.312c45.312 13.056 86.72 40.256 117.376 78.208 5.632 6.784 13.568 10.24 22.08 10.24 6.272 0 12.416-2.24 18.176-6.784 11.904-9.6 13.568-27.84 3.392-39.68-31.808-39.104-72.704-69.12-118.016-87.296zM511.808 391.168c17.024 0 33.408 1.216 49.856 3.456l47.68-49.856c-31.744-6.848-64.064-10.24-97.536-10.24-142.784 0-277.12 63.488-367.232 174.656-10.24 11.904-8.576 30.08 3.904 39.68 5.12 4.48 11.328 6.784 18.176 6.784 7.936 0 15.872-3.968 21.568-10.816 79.872-97.536 197.76-153.664 323.584-153.664zM751.616 400.32l-40.256 41.92c47.04 24.96 89.536 60.032 124.096 102.592 10.24 12.48 27.84 14.208 40.256 3.968 11.968-9.6 13.632-27.84 3.968-39.68-36.16-44.8-79.872-81.088-128.064-108.8zM705.152 244.928l42.56-44.672c-73.664-28.992-153.6-44.224-235.904-44.224-196.672 0-380.864 87.872-505.6 239.744-9.6 12.48-7.872 30.08 3.968 40.256 5.632 3.968 11.904 6.208 18.112 6.208 7.936 0 16.448-3.392 22.144-10.176C163.84 292.608 332.096 212.672 511.808 212.672c66.944 0 132.16 10.752 193.344 32.256zM1017.472 395.776c-40.192-49.92-87.296-92.416-139.456-126.976l-39.68 41.344C889.408 343.04 935.36 383.808 973.888 432c9.6 11.904 27.776 13.568 39.68 3.968 11.84-10.176 14.144-27.712 3.904-40.192zM937.408 104.512c-11.328-10.944-29.312-10.496-40.064 0.832L179.008 854.72c-10.816 11.328-10.496 29.248 0.896 40.064 5.44 5.312 12.48 7.872 19.584 7.872 7.488 0 14.848-2.88 20.416-8.704L938.24 144.576c10.88-11.328 10.496-29.248-0.832-40.064z"></path>
          </svg>
          `;
          NetDiskCheckLinkValidity.setViewAgainCheckClickEvent(ele, checkInfo);
        },
      },
      failed: {
        code: 0,
        msg: "该链接已失效",
        setView(ele, checkInfo) {
          NetDiskCheckLinkValidity.setViewCheckValid(ele, "failed");
          ele.innerHTML = `
          <svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg">
            <path
                  fill="currentColor"
                  d="M549.044706 512l166.189176-166.249412a26.383059 26.383059 0 0 0 0-36.98447 26.383059 26.383059 0 0 0-37.044706 0L512 475.015529l-166.249412-166.249411a26.383059 26.383059 0 0 0-36.98447 0 26.383059 26.383059 0 0 0 0 37.044706L475.015529 512l-166.249411 166.249412a26.383059 26.383059 0 0 0 0 36.98447 26.383059 26.383059 0 0 0 37.044706 0L512 548.984471l166.249412 166.249411a26.383059 26.383059 0 0 0 36.98447 0 26.383059 26.383059 0 0 0 0-37.044706L548.984471 512zM512 1024a512 512 0 1 1 0-1024 512 512 0 0 1 0 1024z"></path>
          </svg>
          `;
          NetDiskCheckLinkValidity.setViewAgainCheckClickEvent(ele, checkInfo);
        },
      },
      needAccessCode: {
        code: 201,
        msg: "该链接缺失提取码",
        setView(ele, checkInfo) {
          NetDiskCheckLinkValidity.setViewCheckValid(ele, "needAccessCode");
          ele.innerHTML = `
          <svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg">
            <path
              fill="currentColor"
              d="M757.810429 373.751333 325.645708 373.751333l0-83.895759c0-103.694687 81.507362-184.922686 185.559183-184.922686 78.121242 0 146.053424 46.74565 173.062568 119.090329 3.865028 10.352789 15.384385 15.609513 25.742291 11.746532 10.351766-3.866051 15.609513-15.390525 11.744485-25.742291C688.844707 121.877815 606.198405 64.918545 511.204891 64.918545c-61.918211 0-119.246895 23.662933-161.423483 66.63156-41.3692 42.142819-64.151066 98.363262-64.151066 158.305469l0 83.895759-20.007683 0c-60.774155 0-110.042255 49.267077-110.042255 110.042255l0 366.139981c0 60.774155 49.267077 110.042255 110.042255 110.042255l492.187769 0c60.775178 0 110.042255-49.267077 110.042255-110.042255L867.852684 483.793588C867.852684 423.01841 818.585607 373.751333 757.810429 373.751333zM827.837318 849.933569c0 38.674834-31.352055 70.02689-70.02689 70.02689L265.62266 919.960459c-38.674834 0-70.02689-31.352055-70.02689-70.02689L195.59577 483.793588c0-38.674834 31.352055-70.02689 70.02689-70.02689l492.187769 0c38.674834 0 70.02689 31.352055 70.02689 70.02689L827.837318 849.933569z"></path>
            <path
              fill="currentColor"
              d="M509.715981 583.832002c-11.048637 0-20.007683 8.959046-20.007683 20.007683l0 110.042255c0 11.048637 8.958022 20.007683 20.007683 20.007683s20.007683-8.958022 20.007683-20.007683L529.723663 603.839685C529.723663 592.790024 520.765641 583.832002 509.715981 583.832002z"></path>
          </svg>
          `;
          NetDiskCheckLinkValidity.setViewAgainCheckClickEvent(ele, checkInfo);
        },
      },
      unknown: {
        code: -200,
        msg: "未知检查情况",
        setView(ele, checkInfo) {
          NetDiskCheckLinkValidity.setViewCheckValid(ele, "unknown");
          ele.innerHTML = `
          <svg viewBox="0 0 1025 1024" xmlns="http://www.w3.org/2000/svg">
            <path
              fill="currentColor"
              d="M512.473172 1023.995242A511.814852 511.814852 0 0 1 313.545134 40.351073a512.244696 512.244696 0 0 1 398.855715 943.658633 508.815937 508.815937 0 0 1-199.927677 39.985536z m0-943.658634C274.559237 80.336608 80.629391 274.266455 80.629391 512.18039s193.929846 431.843781 431.843781 431.843781 431.843781-193.929846 431.843781-431.843781S751.386745 80.336608 512.473172 80.336608z"></path>
            <path
              fill="currentColor"
              d="M506.475342 716.10662a39.985535 39.985535 0 0 1-39.985536-39.985535v-76.972156c0-79.971071 64.976495-144.947566 144.947566-144.947565a77.971794 77.971794 0 0 0 0-155.943588H445.4974a56.979388 56.979388 0 0 0-56.979387 56.979388 39.985535 39.985535 0 0 1-79.971071 0c0-74.972879 60.977941-136.950458 136.950458-136.950459h164.940333c86.968539 0 157.942864 70.974325 157.942865 157.942865s-69.974687 157.942864-157.942865 157.942864a64.976495 64.976495 0 0 0-64.976494 64.976495v76.972156a39.985535 39.985535 0 0 1-38.985897 39.985535zM505.475703 742.097218a48.982281 48.982281 0 1 0 48.982281 48.982281 48.982281 48.982281 0 0 0-48.982281-48.982281z"></path>
          </svg>
          `;
          NetDiskCheckLinkValidity.setViewAgainCheckClickEvent(ele, checkInfo);
        },
      },
    },
    netDisk: {
      baidu: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let url = NetDiskParse.getBlankUrl(
            "baidu",
            netDiskIndex,
            shareCode,
            accessCode
          );
          let getResp = await httpx.get(url, {
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Host: "pan.baidu.com",
              Referer: url,
              Origin: "https://pan.baidu.com",
            },
            onerror() {},
            ontimeout() {},
          });
          let responseText = getResp.data.responseText;
          if (!getResp.status) {
            if (utils.isNull(responseText)) {
              return NetDiskCheckLinkValidity.status.error;
            }
          }
          if (getResp.data.finalUrl.includes("404.html")) {
            return NetDiskCheckLinkValidity.status.error;
          }
          if (responseText.includes("过期时间:")) {
            return NetDiskCheckLinkValidity.status.success;
          } else if (responseText.includes("输入提取")) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          } else if (
            responseText.includes("不存在") ||
            responseText.includes("已失效")
          ) {
            return NetDiskCheckLinkValidity.status.failed;
          } else {
            return NetDiskCheckLinkValidity.status.unknown;
          }
        },
      },
      lanzou: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let url = NetDiskParse.getBlankUrl(
            "lanzou",
            netDiskIndex,
            shareCode,
            accessCode
          );
          let urlObj = new URL(url);

          let getResp = await httpx.get(url, {
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Host: urlObj.hostname,
              Origin: urlObj.origin,
              Referer: url,
            },
            onerror() {},
            ontimeout() {},
          });
          if (!getResp.status) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = getResp.data.responseText;
          if (utils.isNull(data)) {
            return NetDiskCheckLinkValidity.status.failed;
          } else if (data.includes("输入密码")) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          } else if (data.includes("来晚啦") || data.includes("不存在")) {
            return NetDiskCheckLinkValidity.status.failed;
          } else {
            return NetDiskCheckLinkValidity.status.success;
          }
        },
      },
      lanzouyx: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let LanZouYX = new NetDiskParse.netDisk.lanzouyx();
          LanZouYX.uuid = LanZouYX.getEncodeUUID();
          LanZouYX.shareCodeId = LanZouYX.getDecodeShareCodeId(shareCode);
          let devType = 3;
          let devModel = "Chrome";
          let extra = 2;
          let timestamp = LanZouYX.getEncodeTimeStamp();
          let type = 0;

          let offset = 1;
          let limit = 60;
          let postResp = await httpx.post(
            `https://api.ilanzou.com/unproved/recommend/list?devType=${devType}&devModel=${devModel}&uuid=${LanZouYX.uuid}&extra=${extra}&timestamp=${timestamp}&shareId=${LanZouYX.shareCodeId}&type=${type}&offset=${offset}&limit=${limit}`,
            {
              headers: {
                Accept: "application/json, text/plain, */*",
                Origin: "https://www.ilanzou.com",
                Referer: "https://www.ilanzou.com/",
                "Sec-Fetch-Site": "same-site",
                Host: "api.ilanzou.com",
                "User-Agent": utils.getRandomPCUA(),
              },
              responseType: "json",
              onerror() {},
              ontimeout() {},
            }
          );
          if (!postResp.status) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(postResp.data.responseText);
          log.success(["获取链接信息:", data]);
          if (data["code"] !== 200) {
            return NetDiskCheckLinkValidity.status.error;
          }
          if (!data["list"].length) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      tianyiyun: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let postResp = await httpx.post(
            "https://api.cloud.189.cn/open/share/getShareInfoByCodeV2.action",
            {
              data: `shareCode=${shareCode}`,
              headers: {
                Accept: "application/json;charset=UTF-8",
                "Content-Type": "application/x-www-form-urlencoded",
                "User-Agent": utils.getRandomPCUA(),
                "Sign-Type": 1,
                Referer: "https://cloud.189.cn/web/share?code=" + shareCode,
                Origin: "https://cloud.189.cn",
              },
              onerror() {},
              ontimeout() {},
            }
          );
          let responseText = postResp.data.responseText;
          if (!postResp.status && utils.isNull(responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          if (
            responseText.includes("ShareInfoNotFound") ||
            responseText.includes("ShareNotFound") ||
            responseText.includes("FileNotFound") ||
            responseText.includes("ShareAuditWaiting") ||
            responseText.includes("ShareExpiredError") ||
            responseText.includes("ShareAuditNotPass")
          ) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          if (responseText.includes("needAccessCode")) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      hecaiyun: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let resp = await httpx.post(
            "https://caiyun.139.com/stapi/custom/outlink/brief",
            {
              data: "linkId=" + shareCode,
              headers: {
                "Content-Type": "application/x-www-form-urlencoded",
                "User-Agent": utils.getRandomPCUA(),
                Host: "caiyun.139.com",
                Referer: NetDiskParse.getBlankUrl(
                  "hecaiyun",
                  netDiskIndex,
                  shareCode,
                  accessCode
                ),
                Origin: "https://caiyun.139.com",
              },
              ontimeout() {},
              onerror() {},
            }
          );
          if (!resp.status) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(resp.data.responseText);
          if (data.code == 0) {
            if (data.data.isPasswd === "1") {
              return NetDiskCheckLinkValidity.status.needAccessCode;
            } else {
              return NetDiskCheckLinkValidity.status.success;
            }
          } else {
            return NetDiskCheckLinkValidity.status.failed;
          }
        },
      },
      aliyun: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let postResp = await httpx.post(
            "https://api.aliyundrive.com/adrive/v3/share_link/get_share_by_anonymous?share_id=" +
              shareCode,
            {
              data: JSON.stringify({
                share_id: shareCode,
              }),
              headers: {
                Accept: "application/json, text/plain, */*",
                "Content-Type": "application/json",
                "User-Agent": utils.getRandomPCUA(),
                Referer: "https://www.aliyundrive.com/",
                Origin: "https://www.aliyundrive.com",
              },
              onerror() {},
              ontimeout() {},
            }
          );
          let data = utils.toJSON(postResp.data.responseText);
          if (!postResp.status && utils.isNull(data)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          if (
            data["code"] === "ParamFlowException" ||
            data["code"] === "NotFound.ShareLink" ||
            data["code"] === "ShareLink.Cancelled"
          ) {
            return NetDiskCheckLinkValidity.status.failed;
          } else if (
            data["file_count"] === 0 ||
            data["file_infos"]?.length === 0
          ) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      wenshushu: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let postResp = await httpx.post(
            "https://www.wenshushu.cn/ap/task/mgrtask",
            {
              data: JSON.stringify({
                tid: shareCode,
              }),
              headers: {
                "Content-Type": "application/json",
                "User-Agent": utils.getRandomPCUA(),
                "x-token": "wss:7pmakczzw6i",
                Host: "www.wenshushu.cn",
                Origin: "https://www.wenshushu.cn",
                Referer: NetDiskParse.getBlankUrl(
                  "wenshushu",
                  netDiskIndex,
                  shareCode,
                  accessCode
                ),
              },
              responseType: "json",
              ontimeout() {},
              onerror() {},
            }
          );
          if (!postResp.status && utils.isNull(postResp.data.responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(postResp.data.responseText);
          if (data.code !== 0) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      nainiu: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let getResp = await httpx.get(
            "https://cowtransfer.com/core/api/transfer/share?uniqueUrl=" +
              shareCode,
            {
              headers: {
                "User-Agent": utils.getRandomPCUA(),
                Host: "cowtransfer.com",
                Origin: "https://cowtransfer.com/",
                Referer: "https://cowtransfer.com/",
              },
            }
          );
          if (!getResp.status && utils.isNull(getResp.data.responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(getResp.data.responseText);
          if (data.code != "0000") {
            return NetDiskCheckLinkValidity.status.failed;
          }
          if (data.data.needPassword && data.data.needPassword) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      _123pan: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let getResp = await httpx.get(
            "https://www.123pan.com/api/share/info?shareKey=" + shareCode,
            {
              headers: {
                "User-Agent": utils.getRandomPCUA(),
                Host: "www.123pan.com",
                Origin: "https://www.123pan.com/",
                Referer: "https://www.123pan.com/",
              },
              onerror() {},
              ontimeout() {},
              responseType: "json",
            }
          );
          if (!getResp.status && utils.isNull(getResp.data.responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(getResp.data.responseText);
          if (
            getResp.data.responseText.includes("分享页面不存在") ||
            data["code"] !== 0
          ) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          if (data["data"]["HasPwd"]) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      weiyun: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let url = NetDiskParse.getBlankUrl(
            "weiyun",
            netDiskIndex,
            shareCode,
            accessCode
          );
          let getResp = await httpx.get(url, {
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Host: "share.weiyun.com",
              Origin: "https://share.weiyun.com",
              Referer: url,
            },
            onerror() {},
            ontimeout() {},
          });

          if (!getResp.status && utils.isNull(getResp.data.responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let respText = getResp.data.responseText;
          if (
            respText.includes("已删除") ||
            respText.includes("违反相关法规") ||
            respText.includes("已过期") ||
            respText.includes("已经删除") ||
            respText.includes("目录无效")
          ) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          if (
            respText.includes('"need_pwd":1') ||
            respText.includes('"pwd":"')
          ) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      xunlei: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let postResp = await httpx.post(
            "https://xluser-ssl.xunlei.com/v1/shield/captcha/init",
            {
              data: JSON.stringify({
                client_id: "Xqp0kJBXWhwaTpB6",
                device_id: "925b7631473a13716b791d7f28289cad",
                action: "get:/drive/v1/share",
                meta: {
                  package_name: "pan.xunlei.com",
                  client_version: "1.45.0",
                  captcha_sign: "1.fe2108ad808a74c9ac0243309242726c",
                  timestamp: "1645241033384",
                },
              }),
              headers: {
                "User-Agent": utils.getRandomPCUA(),
                Host: "pan.xunlei.com",
                Referer: NetDiskParse.getBlankUrl(
                  "xunlei",
                  netDiskIndex,
                  shareCode,
                  accessCode
                ),
                Origin: "https://pan.xunlei.com",
              },
              onerror() {},
              ontimeout() {},
            }
          );
          if (!postResp.status && utils.isNull(postResp.data.responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(postResp.data.responseText);
          let token = data["captcha_token"];
          let getResp = await httpx.get(
            "https://api-pan.xunlei.com/drive/v1/share?share_id=" + shareCode,
            {
              headers: {
                "User-Agent": utils.getRandomPCUA(),
                Host: "pan.xunlei.com",
                Referer: NetDiskParse.getBlankUrl(
                  "xunlei",
                  netDiskIndex,
                  shareCode,
                  accessCode
                ),
                Origin: "https://pan.xunlei.com",
                "x-captcha-token": token,
                "x-client-id": "Xqp0kJBXWhwaTpB6",
                "x-device-id": "925b7631473a13716b791d7f28289cad",
              },
              onerror() {},
              ontimeout() {},
            }
          );

          if (!getResp.status && utils.isNull(getResp.data.responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let responseText = getResp.data.responseText;
          if (
            responseText.includes("NOT_FOUND") ||
            responseText.includes("SENSITIVE_RESOURCE") ||
            responseText.includes("EXPIRED") ||
            responseText.includes("DELETED")
          ) {
            return NetDiskCheckLinkValidity.status.failed;
          } else if (responseText.includes("PASS_CODE_EMPTY")) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      _115pan: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let getResp = await httpx.get(
            `https://webapi.115.com/share/snap?share_code=${shareCode}&offset=0&limit=20&receive_code=&cid=`,
            {
              headers: {
                Accept: "application/json, text/javascript, */*;",
                "User-Agent": utils.getRandomPCUA(),
                Host: "webapi.115.com",
                Referer: "https://115.com/",
                Origin: "https://115.com",
              },
              ontimeout() {},
              onerror() {},
            }
          );
          if (!getResp.status) {
            if (utils.isNull(getResp.data.responseText)) {
              return NetDiskCheckLinkValidity.status.failed;
            }
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(getResp.data.responseText);
          if (data.state) {
            return NetDiskCheckLinkValidity.status.success;
          }
          if (typeof data.error === "string") {
            if (data.error.includes("访问码")) {
              return NetDiskCheckLinkValidity.status.needAccessCode;
            } else if (
              data.error.includes("链接") ||
              data.error.includes("分享已取消")
            ) {
              return NetDiskCheckLinkValidity.status.failed;
            }
          }

          return NetDiskCheckLinkValidity.status.unknown;
        },
      },
      chengtong: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          /* 城通通用的检查api */
          /* ref是来源 */
          let blankUrl = NetDiskParse.getBlankUrl(
            "chengtong",
            netDiskIndex,
            shareCode,
            accessCode
          );
          let blankUrlObj = new URL(blankUrl);
          /* f是文件 d是文件夹 */
          const path = blankUrlObj.pathname.split("/")[1].trim();
          let url = "";
          if (path === "f" || path === "file") {
            /* 文件 */
            url = `https://webapi.ctfile.com/getfile.php?path=${path}&f=${shareCode}&passcode=${accessCode}&token=0&r=${Math.random()}&ref=`;
          } else if (path === "d" || path === "dir") {
            /* 文件夹 */
            url = `https://webapi.ctfile.com/getdir.php?path=${path}&d=${shareCode}&folder_id=&passcode=${accessCode}&token=0&r=${Math.random()}&ref=`;
          } else {
            log.warn("未知path", [netDiskIndex, shareCode, accessCode]);
            return NetDiskCheckLinkValidity.status.unknown;
          }
          let getResp = await httpx.get(url, {
            headers: {
              Host: "webapi.ctfile.com",
              Origin: "https://url95.ctfile.com",
              Referer: blankUrl,
              Accept: "application/json, text/javascript, */*; q=0.01",
              "User-Agent": utils.getRandomPCUA(),
            },
            ontimeout() {},
            onerror() {},
          });
          let responseText = getResp.data.responseText;
          if (!getResp.status && utils.isNull(responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(responseText);
          if (data["code"] === 200) {
            return NetDiskCheckLinkValidity.status.success;
          }
          if (data["code"] === 401) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          }
          if (data["code"] === 404 || data["code"] === 503) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          return NetDiskCheckLinkValidity.status.unknown;
        },
      },
      kuake: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let url =
            "https://drive.quark.cn/1/clouddrive/share/sharepage/token?pr=ucpro&fr=pc";
          let postResp = await httpx.post(url, {
            data: JSON.stringify({
              pwd_id: shareCode,
              passcode: "",
            }),
            headers: {
              Accept: "application/json, text/plain, */*",
              "Content-Type": "application/json;charset=UTF-8",
              "User-Agent": utils.getRandomPCUA(),
              Origin: "https://pan.quark.cn",
              Referer: NetDiskParse.getBlankUrl(
                "kuake",
                netDiskIndex,
                shareCode,
                accessCode
              ),
            },
            onerror() {},
            ontimeout() {},
          });
          if (!postResp.status && utils.isNull(postResp.data.responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(postResp.data.responseText);
          if (data.message.includes("需要提取码")) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          } else if (data.message.includes("ok")) {
            return NetDiskCheckLinkValidity.status.success;
          } else {
            return NetDiskCheckLinkValidity.status.failed;
          }
        },
      },
      jianguoyun: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let url = NetDiskParse.getBlankUrl(
            "jianguoyun",
            netDiskIndex,
            shareCode,
            accessCode
          );
          let getResp = await httpx.get(url, {
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Host: "www.jianguoyun.com",
              Referer: NetDiskParse.getBlankUrl(
                "jianguoyun",
                netDiskIndex,
                shareCode,
                accessCode
              ),
              Origin: "https://www.jianguoyun.com",
            },
            onerror() {},
            ontimeout() {},
          });
          let responseText = getResp.data.responseText;
          if (!getResp.status && utils.isNull(responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          if (responseText.includes("<h1>啊噢!")) {
            return NetDiskCheckLinkValidity.status.failed;
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      onedrive: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let url = NetDiskParse.getBlankUrl(
            "onedrive",
            netDiskIndex,
            shareCode,
            accessCode
          );
          let urlObj = new URL(url);
          let getResp = await httpx.get(url, {
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Host: urlObj.hostname,
              Referer: url,
              Origin: urlObj.origin,
            },
            onerror() {},
            ontimeout() {},
          });
          if (!getResp.status) {
            let status = getResp.data?.status?.toString();
            if (status === "429") {
              return NetDiskCheckLinkValidity.status.error;
            } else if (status === "404") {
              return NetDiskCheckLinkValidity.status.failed;
            }
            return NetDiskCheckLinkValidity.status.error;
          }
          let responseText = getResp.data.responseText;
          if (utils.isNotNull(responseText)) {
            try {
              let respDOM = DOMUtils.parseHTML(responseText, true, true);
              if (respDOM.querySelector("title")?.innerHTML?.includes("错误")) {
                return NetDiskCheckLinkValidity.status.failed;
              }
            } catch (error) {}
          }
          return NetDiskCheckLinkValidity.status.success;
        },
      },
      uc: {
        /**
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode 分享码
         * @param {string} accessCode 访问码
         */
        async init(netDiskIndex, shareCode, accessCode) {
          let postResp = await httpx.post(
            "https://pc-api.uc.cn/1/clouddrive/share/sharepage/token?pr=UCBrowser&fr=pc",
            {
              data: JSON.stringify({ pwd_id: shareCode, passcode: accessCode }),
              headers: {
                "User-Agent": utils.getRandomPCUA(),
                Host: "drive.uc.cn",
                Referer: NetDiskParse.getBlankUrl(
                  "uc",
                  netDiskIndex,
                  shareCode,
                  accessCode
                ),
                Origin: "https://drive.uc.cn",
              },
              ontimeout() {},
              onerror() {},
            }
          );
          let responseText = postResp.data.responseText;
          if (!postResp.status && utils.isNull(responseText)) {
            return NetDiskCheckLinkValidity.status.error;
          }
          let data = utils.toJSON(responseText);
          if (data["status"] === 200) {
            return NetDiskCheckLinkValidity.status.success;
          }
          if (data["message"].includes("需要提取码")) {
            return NetDiskCheckLinkValidity.status.needAccessCode;
          }
          if (data["message"].includes("inner error")) {
            return NetDiskCheckLinkValidity.status.error;
          }
          return NetDiskCheckLinkValidity.status.failed;
        },
      },
    },
    /**
     * 校验链接的有效性,这里是用于订阅-消费
     * @param {HTMLDivElement} netDiskViewBox
     * @param {string} netDiskName
     * @param {number} netDiskIndex
     * @param {string} shareCode
     * @param {string} accessCode
     */
    async check(
      netDiskViewBox,
      netDiskName,
      netDiskIndex,
      shareCode,
      accessCode
    ) {
      this.$data.subscribe.push({
        netDiskViewBox,
        netDiskName,
        netDiskIndex,
        shareCode,
        accessCode,
      });

      if (this.$data.isSubscribing) {
        return;
      }
      this.$data.isSubscribing = true;
      for (let index = 0; index < this.$data.subscribe.length; index++) {
        const checkInfo = this.$data.subscribe[index];
        await this.checkLinkValidity(checkInfo, true);
        await utils.sleep(250);
        this.$data.subscribe.splice(index, 1);
        index--;
      }
      this.$data.isSubscribing = false;
    },
    /**
     * 开始校验链接的有效性
     * @param {NetDiskCheckLinkValidityOption} checkInfo
     * @param {boolean} isForceCheck 是否强制检测
     */
    async checkLinkValidity(checkInfo, isForceCheck) {
      let $netDiskStatus =
        checkInfo.netDiskViewBox.querySelector(".netdisk-status");
      if (this.isViewCheckValid($netDiskStatus) && !isForceCheck) {
        return;
      }
      let regular =
        NetDisk.regular[checkInfo.netDiskName][checkInfo.netDiskIndex];
      if (!regular["checkLinkValidity"]) {
        log.error(["未配置checkLinkValidity", checkInfo]);
        return;
      }
      let netDiskCheck = this.netDisk[checkInfo.netDiskName];
      if (
        !netDiskCheck ||
        (netDiskCheck && typeof netDiskCheck.init !== "function")
      ) {
        /* 没有配置该网盘的校验有效性 */
        log.error(["没有配置该网盘的校验有效性", checkInfo]);
        return;
      }
      this.status.loading.setView($netDiskStatus);
      /**
       * @type {NetDiskCheckLinkValidityStatus}
       */
      let result = await netDiskCheck.init(
        checkInfo.netDiskIndex,
        checkInfo.shareCode,
        checkInfo.accessCode
      );
      if (!result) {
        log.error(["该验证函数的返回值不是有效值", [result, checkInfo]]);
        return;
      }
      result.setView($netDiskStatus, checkInfo);
    },
    /**
     * 添加重复检查
     * @param {HTMLDivElement} ele
     * @param {NetDiskCheckLinkValidityOption} checkInfo
     */
    setViewAgainCheckClickEvent(ele, checkInfo) {
      DOMUtils.on(
        ele,
        "click",
        void 0,
        function () {
          const $netDiskUrlDiv = ele.closest(".netdisk-url-div");
          const $netDiskLink = $netDiskUrlDiv.querySelector(".netdisk-link");
          const linkData = NetDiskUI.getElementNetDiskLinkData($netDiskLink);
          linkData["netDiskViewBox"] = checkInfo.netDiskViewBox;
          NetDiskCheckLinkValidity.checkLinkValidity(linkData, true);
        },
        { once: true }
      );
    },
    /**
     * 判断元素当前是否处于验证状态且验证是error或未验证状态
     * + true 已验证(成功/需要密码)
     * + false 尚未验证/验证超时/验证网络异常
     * @param {HTMLDivElement} ele
     */
    isViewCheckValid(ele) {
      if (!ele.hasAttribute("data-check-valid")) {
        return false;
      }
      if (ele.getAttribute("data-check-valid") === "error") {
        return false;
      }
      return true;
    },
    /**
     * 设置当前的验证状态
     * @param {HTMLDivElement} ele
     * @param {string} value
     */
    setViewCheckValid(ele, value) {
      ele.setAttribute("data-check-valid", value);
    },
    /**
     * 取消设置当前的验证状态
     * @param {HTMLDivElement} ele
     */
    removeViewCheckValid(ele) {
      ele.removeAttribute("data-check-valid");
    },
    /**
     * 判断状态码是成功的
     * @param {NetDiskCheckLinkValidityStatus} statusInfo
     */
    isStatusSuccess(statusInfo) {
      if (Math.floor(statusInfo.code / 100) === 2) {
        return true;
      }
      return false;
    },
    /**
     * 根据code获取code的名字
     * @param {NetDiskCheckLinkValidityStatus} statusInfo
     */
    getStatusName(statusInfo) {
      for (const statusName of Object.keys(NetDiskCheckLinkValidity.status)) {
        let _statusInfo_ = NetDiskCheckLinkValidity.status[statusName];
        if (statusInfo.code === _statusInfo_.code) {
          return statusName;
        }
      }
    },
  };

  /** 网盘-直链解析 */
  const NetDiskParse = {
    netDisk: {
      /**
       * 百度网盘
       * @constructor
       * @returns {object}
       */
      baidu: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        /**
         * 入口
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode
         * @param {string} accessCode
         * @returns
         */
        this.init = function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          let url = GM_getValue("baidu-baiduwp-php-url");
          let postForm = GM_getValue("baidu-baiduwp-php-post-form");
          let enableCopy = GM_getValue("baidu-baiduwp-php-copy-url");
          if (!url) {
            Qmsg.error("请先在设置中配置百度网盘-网址");
            return void 0;
          }
          if (!postForm) {
            Qmsg.error("请先在设置中配置百度网盘-表单参数");
            return void 0;
          }
          postForm = NetDisk.replaceParam(postForm, {
            shareCode: shareCode,
            accessCode: accessCode,
          });
          let formElement = document.createElement("form");
          /* POST的表单数据 */
          let formData = {};
          let urlParams = new URLSearchParams(postForm);
          /* 解析网址 */
          formElement.action = url;
          formElement.method = "post";
          formElement.style.display = "none";
          formElement.target = "_blank";
          for (let [key, value] of urlParams) {
            let textAreaElement = document.createElement("textarea");
            textAreaElement.name = key;
            textAreaElement.value = value;
            formElement.appendChild(textAreaElement);
            formData[key] = value;
          }
          log.info(["表单数据", formData]);
          document.body.appendChild(formElement);
          log.info(["访问网址", url]);
          if (enableCopy) {
            NetDiskParse.copyText(
              "baidu",
              netDiskIndex,
              shareCode,
              accessCode,
              "1.5秒后跳转至解析站"
            );
            setTimeout(() => {
              formElement.submit();
            }, 1500);
          } else {
            formElement.submit();
          }
        };
        return this;
      },
      /**
       * 蓝奏云
       * 流程:判断是否是多文件
       * 单文件 => 请求https://蓝奏云域名/{shareToken} 判断链接类型和是否能正常获取
       *        => 请求https://蓝奏云域名/ajaxm.php 获取下载参数,下载参数例如:https://蓝奏云文件域名/file/?xxxxxxxxx
       * 多文件 => 先请求https://蓝奏云域名/{shareToken} 获取文件sign => 请求https://蓝奏云域名/filemoreajax.php 获取json格式的文件参数,
       * 参数内容如{"info":"success","text":[{"duan":"xx","icon":"","id":"".....},{},{}]}
       * @constructor
       * @returns {object}
       */
      lanzou: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        /* 蓝奏云域名 */
        let LanZouHostName = GM_getValue(
          "lanzou-host-name",
          NetDiskData.lanzou_defaultHostName
        );
        /**
         * 路由
         */
        this.router = {
          default(pathName = "") {
            if (pathName.startsWith("/")) {
              pathName = pathName.replace(/^\//, "");
            }
            return `https://${LanZouHostName}/${pathName}`;
          },
          tp(pathName = "") {
            if (pathName.startsWith("/")) {
              pathName = pathName.replace(/^\//, "");
            }
            return `https://${LanZouHostName}/tp/${pathName}`;
          },
          s(pathName = "") {
            if (pathName.startsWith("/")) {
              pathName = pathName.replace(/^\//, "");
            }
            return `https://${LanZouHostName}/s/${pathName}`;
          },
        };
        this.regexp = {
          unicode: {
            /**
             * 判断该链接是否是中文
             */
            match: /[%\u4e00-\u9fa5]+/g,
            tip: "中文链接",
            isUnicode: false,
          },
          /**
           * 蓝奏文件取消分享
           */
          noFile: {
            match: /div>来晚啦...文件取消分享了<\/div>/g,
            tip: "来晚啦...文件取消分享了",
          },
          /**
           * 蓝奏文件链接错误
           */
          noExists: {
            match: /div>文件不存在,或已删除<\/div>/g,
            tip: "文件不存在,或已删除",
          },
          /**
           * 2023-9-19 蓝奏云修改分享规则,需要vip用户才可以分享 apk、ipa 链接
           */
          needVipToShare: {
            match: /class="fbox">非会员.+请先开通会员/gi,
            tip: "该链接为非会员用户分享的文件,目前无法下载",
          },
          /**
           * 蓝奏多文件
           */
          moreFile: {
            match: /<span id=\"filemore\" onclick=\"more\(\);\">/g,
          },
          /**
           * 蓝奏设置了密码的单文件请求需要的sign值
           */
          sign: {
            match:
              /var[\s]*(posign|postsign|vidksek|skdklds)[\s]*=[\s]*'(.+?)';/,
          },
          /**
           * 蓝奏文件名
           */
          fileName: {
            match: /<title>(.*)<\/title>/,
          },
          /**
           * 蓝奏文件大小
           */
          fileSize: {
            match: /<span class=\"mtt\">\((.*)\)<\/span>/,
          },
          /**
           * 蓝奏文件直链host
           */
          loadDownHost: {
            match: /var[\s]*(vkjxld)[\s]*=[\s]*'(.+?)'/i,
          },
          /**
           * 蓝奏文件直链
           */
          loadDown: {
            match:
              /var[\s]*(loaddown|oreferr|spototo|domianload|hyggid)[\s]*=[\s]*'(.+?)'/i,
          },
          /**
           * 蓝奏云之苹果使用类型的文件
           */
          appleDown: {
            match: /var[\s]*appitem[\s]*=[\s]*'(.+?)'/i,
          },
          /**
           * 蓝奏云文件上传时间
           */
          uploadTime: {
            match: /mt2\"\>时间:<\/span>(.+?)[\s]*<span/i,
          },
        };
        /**
         * 入口
         * @param {number} netDiskIndex
         * @param {string} shareCode
         * @param {string} accessCode
         */
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          that.regexp.unicode.isUnicode = Boolean(
            that.shareCode.match(that.regexp.unicode.match)
          );
          if (netDiskIndex === 2) {
            await that.getMoreFile(that.router.s(that.shareCode));
          } else {
            await that.getFileLink();
          }
        };
        /**
         * 获取文件链接
         * @param {boolean} getShareCodeByPageAgain
         * @returns
         */
        this.getFileLink = async function (getShareCodeByPageAgain = false) {
          let url = that.router.default(that.shareCode);
          log.info("蓝奏云-获取文件下载链接" + url);
          let getResp = await httpx.get({
            url: url,
            headers: {
              Accept: "*/*",
              "User-Agent": utils.getRandomPCUA(),
              Referer: url,
            },
            onerror() {},
          });
          if (!getResp.status) {
            log.error(getResp);
            if (getResp.type === "ontimeout") {
              return;
            }
            if (utils.isNull(getResp.data.responseText)) {
              Qmsg.error("请求异常");
              return;
            }
            if (
              getResp.data.responseText.includes(
                "div>文件不存在,或者已被删除</div>"
              )
            ) {
              Qmsg.error("文件不存在,或者已被删除");
            } else {
              Qmsg.error("未知情况");
            }
            return;
          }
          let respData = getResp.data;
          if (respData.readyState !== 4) {
            log.error(respData);
            Qmsg.error("请求失败,请重试");
            return;
          }
          if (respData.responseText == void 0) {
            log.error(respData);
            Qmsg.error("获取网页内容为空");
            return;
          }
          if (!that.checkPageCode(respData)) {
            return;
          }
          if (that.isMoreFile(respData)) {
            await that.getMoreFile();
          } else {
            log.info(respData);
            let pageText = respData.responseText;
            if (getShareCodeByPageAgain) {
              let shareCodeNewMatch = pageText.match(
                /var[\s]*link[\s]*=[\s]*\'tp\/(.+?)\';/i
              );
              that.shareCode = shareCodeNewMatch[shareCodeNewMatch.length - 1];
              log.info(`新参数 => ${that.shareCode}`);
            }
            let pageDOM = DOMUtils.parseHTML(pageText, true, true);
            let pageIframeElement =
              pageDOM.querySelector('iframe[class^="ifr"]') ||
              pageDOM.querySelector('iframe[class^="n_downlink"]');
            if (pageIframeElement) {
              let iframeUrl = pageIframeElement.getAttribute("src");
              log.error([
                "该链接需要重新通过iframe地址访问获取信息",
                iframeUrl,
              ]);
              Qmsg.info("正在请求下载信息");
              let fileName =
                pageDOM.querySelector("body div.d > div")?.innerText ||
                pageDOM.querySelector("#filenajax")?.innerText ||
                pageDOM
                  .querySelector("title")
                  ?.textContent?.replace(/ - 蓝奏云$/i, "");
              let fileSize =
                pageText.match(/文件大小:<\/span>(.+?)<br>/i) ||
                pageDOM.querySelector("div.n_box div.n_file div.n_filesize")
                  ?.innerText ||
                pageDOM.querySelector(
                  ".d2 table tr td .fileinfo:nth-child(1) .fileinforight"
                )?.innerText;
              let fileUploadTime =
                pageText.match(/上传时间:<\/span>(.+?)<br>/i) ||
                pageDOM.querySelector(
                  "#file[class=''] .n_file_info span.n_file_infos"
                )?.innerText ||
                pageDOM.querySelector(
                  ".d2 table tr td .fileinfo:nth-child(3) .fileinforight"
                )?.innerText ||
                pageDOM.querySelector(
                  "#file[class='filter'] .n_file_info span.n_file_infos"
                )?.innerText;
              if (fileSize) {
                if (Array.isArray(fileSize)) {
                  fileSize = fileSize[fileSize.length - 1];
                }
                if (typeof fileSize === "string") {
                  fileSize = fileSize.replaceAll("大小:", "");
                }
              } else {
                log.error("解析文件大小信息失败");
              }
              if (fileUploadTime) {
                if (Array.isArray(fileUploadTime)) {
                  fileUploadTime = fileUploadTime[fileUploadTime.length - 1];
                }
                if (
                  fileUploadTime.toString().toLowerCase().startsWith("android")
                ) {
                  log.error("解析出的文件上传时间信息是Android/xxxx开头");
                  fileUploadTime = void 0;
                }
              } else {
                log.error("解析文件上传时间信息失败");
              }
              let downloadUrl = await that.getLinkByIframe(iframeUrl, {
                fileName,
                fileSize,
                fileUploadTime,
              });
              if (downloadUrl) {
                if (NetDiskFilterScheme.isForwardDownloadLink("lanzou")) {
                  downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                    "lanzou",
                    downloadUrl
                  );
                }
                NetDiskUI.staticView.oneFile({
                  title: "蓝奏云单文件直链",
                  fileName: fileName,
                  fileSize: fileSize,
                  downloadUrl: downloadUrl,
                  fileUploadTime: fileUploadTime,
                });
              }
            } else {
              await that.getLink(respData);
            }
          }
        };
        /**
         * 页面检查,看看是否存在文件失效情况
         * @param {object} response
         * @returns {boolean}
         * + true 未失效
         * + false 已失效
         */
        this.checkPageCode = function (response) {
          let pageText = response.responseText;
          if (pageText.match(that.regexp.noFile.match)) {
            Qmsg.error(that.regexp.noFile.tip);
            return false;
          }
          if (pageText.match(that.regexp.noExists.match)) {
            Qmsg.error(that.regexp.noExists.tip);
            return false;
          }
          if (pageText.match(that.regexp.needVipToShare.match)) {
            Qmsg.error(that.regexp.needVipToShare.tip);
            return false;
          }
          return true;
        };
        /**
         * 判断是否是多文件的链接
         * @param {object} response
         * @returns {boolean}
         * + true 多文件
         * + false 单文件
         */
        this.isMoreFile = function (response) {
          let pageText = response.responseText;
          if (pageText.match(that.regexp.moreFile.match)) {
            log.info("该链接为多文件");
            return true;
          } else {
            log.info("该链接为单文件");
            return false;
          }
        };
        /**
         * 获取链接
         * @param {object} response
         */
        this.getLink = async function (response) {
          let pageText = response.responseText;
          if (pageText == void 0) {
            log.error("shareCode错误,重新从页面中获取");
            await that.getFileLink(true);
            return;
          }
          let sign = pageText.match(that.regexp.sign.match);
          let postData_p = "";
          let postData_sign = "";
          let fileName = pageText.match(that.regexp.fileName.match);
          let fileSize =
            pageText.match(that.regexp.fileSize.match) ||
            pageText.match(/<div class="n_filesize">大小:(.+?)<\/div>/i);
          let fileUploadTime =
            pageText.match(that.regexp.uploadTime.match) ||
            pageText.match(/<span class="n_file_infos">(.+?)<\/span>/i);
          if (fileName) {
            fileName = fileName[fileName.length - 1].trim();
          } else {
            fileName = "";
          }
          if (fileSize) {
            fileSize = fileSize[fileSize.length - 1].trim();
          } else {
            fileSize = "";
          }
          if (fileUploadTime) {
            fileUploadTime = fileUploadTime[fileUploadTime.length - 1].trim();
          } else {
            fileUploadTime;
          }
          if (sign) {
            postData_sign = sign[sign.length - 1];
            log.info(`获取Sign: ${postData_sign}`);
            if (utils.isNotNull(that.accessCode)) {
              log.info("传入参数=>有密码");
              postData_p = that.accessCode;
            } else {
              log.info("传入参数=>无密码");
            }
            let postResp = await httpx.post({
              url: that.router.default("ajaxm.php"),
              responseType: "json",
              headers: {
                "Content-Type":
                  "application/x-www-form-urlencoded; charset=UTF-8",
                "User-Agent": utils.getRandomAndroidUA(),
                Referer: that.router.default(that.shareCode),
              },
              data: `action=downprocess&sign=${postData_sign}&p=${postData_p}`,
            });
            if (!postResp.status) {
              return;
            }
            let respData = postResp.data;
            log.info(respData);
            if (respData.readyState === 4) {
              let json_data = utils.toJSON(respData.responseText);
              let downloadUrl = `${json_data["dom"]}/file/${json_data["url"]}`;
              if (
                typeof json_data["url"] === "string" &&
                (json_data["url"].startsWith("http") ||
                  json_data["url"].startsWith(json_data["dom"]))
              ) {
                /* 有些情况下比如苹果的ipa文件的请求,json_data["url"]就是一个完整的链接 */
                downloadUrl = json_data["url"];
              }
              /* json_data["zt"]表示状态,一般为1 */
              let zt = json_data["zt"];
              /* json_data["inf"]一般是文件名,也可以看作是请求信息提示 */
              if ("密码不正确".indexOf(json_data["inf"]) != -1) {
                Qmsg.error("密码不正确!");
                NetDiskUI.newAccessCodeView(
                  void 0,
                  "lanzou",
                  that.netDiskIndex,
                  that.shareCode,
                  (userInputAccessCode) => {
                    that.init(
                      that.netDiskIndex,
                      that.shareCode,
                      userInputAccessCode
                    );
                  }
                );
              } else {
                fileName = json_data["inf"] ? json_data["inf"] : fileName;
                downloadUrl = await NetDiskParse.getRedirectFinalUrl(
                  downloadUrl,
                  utils.getRandomAndroidUA()
                );
                log.info(downloadUrl);

                if (NetDiskFilterScheme.isForwardDownloadLink("lanzou")) {
                  downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                    "lanzou",
                    downloadUrl
                  );
                }

                NetDiskUI.staticView.oneFile({
                  title: "蓝奏云单文件直链",
                  fileName: fileName,
                  fileSize: fileSize,
                  downloadUrl: downloadUrl,
                  fileUploadTime: fileUploadTime,
                });
              }
            } else {
              Qmsg.error("请求失败,请重试");
            }
          } else {
            let loadDownHost = pageText.match(that.regexp.loadDownHost.match);
            let loadDown = pageText.match(that.regexp.loadDown.match);
            let appleDown = pageText.match(that.regexp.appleDown.match);
            if (utils.isNull(loadDown)) {
              loadDown = pageText.match(/var[\s]*(cppat)[\s]*=[\s]*'(.+?)'/i);
            }
            if (utils.isNull(loadDownHost) && appleDown) {
              appleDown = appleDown[appleDown.length - 1];
              loadDownHost = [appleDown];
              loadDown = [""];
              log.success(["多文件-当前链接猜测为苹果的文件", appleDown]);
            }
            if (utils.isNull(loadDownHost)) {
              Qmsg.error("蓝奏云直链:获取sign的域名失败,请反馈开发者", {
                timeout: 3500,
              });
              return;
            }
            if (utils.isNull(loadDown)) {
              Qmsg.error("蓝奏云直链:获取sign失败,请反馈开发者", {
                timeout: 3500,
              });
              return;
            }
            let downloadUrl = `${loadDownHost[loadDownHost.length - 1]}${
              loadDown[loadDown.length - 1]
            }`;
            log.info([fileName, fileSize, downloadUrl]);
            downloadUrl = await NetDiskParse.getRedirectFinalUrl(
              downloadUrl,
              utils.getRandomAndroidUA()
            );
            log.info(downloadUrl);
            if (NetDiskFilterScheme.isForwardDownloadLink("lanzou")) {
              downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                "lanzou",
                downloadUrl
              );
            }

            NetDiskUI.staticView.oneFile({
              title: "蓝奏云单文件直链",
              fileName: fileName,
              fileSize: fileSize,
              downloadUrl: downloadUrl,
              fileUploadTime: fileUploadTime,
            });
          }
        };

        /**
         * 通过iframe的链接来获取单文件直链
         * @param {string} urlPathName url路径
         * @param {{
         * fileName:string,
         * fileSize:string,
         * fileUploadTime:string
         * }} fileInfo 文件信息
         */
        this.getLinkByIframe = async function (urlPathName, fileInfo) {
          log.info([urlPathName, fileInfo]);
          let iFrameUrl = that.router.default(urlPathName);
          let getResp = await httpx.get({
            url: iFrameUrl,
            headers: {
              Accept:
                "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7",
              "User-Agent": utils.getRandomPCUA(),
              Referer: that.router.default(that.shareCode),
            },
          });
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          log.info(respData);
          let pageText = respData.responseText;
          let aihidcms = pageText.match(/var[\s]*aihidcms[\s]*=[\s]*'(.*)';/i);
          let ciucjdsdc = pageText.match(
            /var[\s]*ciucjdsdc[\s]*=[\s]*'(.*)';/i
          );
          let ajaxdata = pageText.match(/var[\s]*ajaxdata[\s]*=[\s]*'(.+)';/i);
          let sign = pageText.match(/'sign':[\s]*'(.+)',/i);
          let ajaxUrl = pageText.match(/url[\s]*:[\s]*'(.+)'[\s]*,/);
          if (ajaxUrl) {
            ajaxUrl = ajaxUrl[ajaxUrl.length - 1];
          } else {
            Qmsg.error("提取ajaxm.php的具体参数失败,使用默认的ajaxm.php");
            ajaxUrl = "ajaxm.php";
          }
          if (!aihidcms) {
            Qmsg.error("ajaxm.php请求参数 websignkey 获取失败");
            return;
          } else {
            aihidcms = aihidcms[aihidcms.length - 1];
          }
          if (!ciucjdsdc) {
            Qmsg.error("ajaxm.php请求参数 websign 获取失败");
            return;
          } else {
            ciucjdsdc = ciucjdsdc[ciucjdsdc.length - 1];
          }
          if (!ajaxdata) {
            Qmsg.error("ajaxm.php请求参数 signs 获取失败");
            return;
          } else {
            ajaxdata = ajaxdata[ajaxdata.length - 1];
          }
          if (!sign) {
            Qmsg.error("ajaxm.php请求参数 sign 获取失败");
            return;
          } else {
            sign = sign[sign.length - 1];
          }
          let postData = `action=downprocess&signs=${ajaxdata}&sign=${sign}&websign=${ciucjdsdc}&ves=1&websignkey=${aihidcms}`;
          log.success("请求的路径参数:" + ajaxUrl);
          log.success("ajaxm.php的请求参数-> " + postData);
          let postResp = await httpx.post({
            url: that.router.default(ajaxUrl),
            headers: {
              Accept: "application/json, text/javascript, */*",
              "Content-Type": "application/x-www-form-urlencoded",
              Referer: that.router.default(that.shareCode),
              "User-Agent": utils.getRandomPCUA(),
            },
            data: postData,
          });
          if (!postResp.status) {
            return;
          }
          let postRespData = postResp.data;
          log.info(postRespData);
          let jsonData = utils.toJSON(postRespData.responseText);
          let downloadUrl = `${jsonData["dom"]}/file/${jsonData["url"]}`;
          let zt = jsonData["zt"];
          log.success(["直链", downloadUrl]);
          if ("密码不正确".indexOf(jsonData["inf"]) != -1) {
            Qmsg.error("密码不正确!");
            NetDiskUI.newAccessCodeView(
              void 0,
              "lanzou",
              that.netDiskIndex,
              that.shareCode,
              (userInputAccessCode) => {
                that.init(
                  that.netDiskIndex,
                  that.shareCode,
                  userInputAccessCode
                );
              }
            );
          } else {
            fileInfo.fileName = utils.isNotNull(jsonData["inf"])
              ? jsonData["inf"]
              : fileInfo.fileName;
            downloadUrl = await NetDiskParse.getRedirectFinalUrl(
              downloadUrl,
              utils.getRandomAndroidUA()
            );
            log.info(downloadUrl);
            return downloadUrl;
          }
        };
        /**
         * 多文件获取
         * @param {string} url 链接
         */
        this.getMoreFile = async function (url) {
          if (url == void 0) {
            url = that.router.default(that.shareCode);
          }
          let getResp = await httpx.get({
            url: url,
            headers: {
              Accept: "*/*",
              "User-Agent": utils.getRandomAndroidUA(),
              Referer: url,
            },
          });
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          log.info(respData);
          if (respData.readyState !== 4) {
            Qmsg.error("请求失败,请重试");
            return;
          }
          let pageText = respData.responseText;
          let fid = pageText.match(/\'fid\':(.+?),/)[1].replaceAll("'", "");
          let uid = pageText.match(/\'uid\':(.+?),/)[1].replaceAll("'", "");
          let pgs = 1;
          let t_name = pageText.match(/\'t\':(.+?),/)[1];
          let t_rexp = new RegExp(t_name + "[\\s]*=[\\s]*('|\")(.+?)('|\");");
          let t = pageText.match(t_rexp)[2];
          let k_name = pageText.match(/\'k\':(.+?),/)[1];
          let k_rexp = new RegExp(k_name + "[\\s]*=[\\s]*('|\")(.+?)('|\");");
          let k = pageText.match(k_rexp)[2];
          let lx = that.shareCode.match(that.regexp.unicode.match) ? 1 : 2;
          let postData = `lx=${lx}&fid=${fid}&uid=${uid}&pg=${pgs}&rep=0&t=${t}&k=${k}&up=1&ls=1&pwd=${that.accessCode}`;
          log.info(`多文件请求参数:${postData}`);
          let postResp = await httpx.post({
            url: that.router.default("filemoreajax.php"),
            responseType: "json",
            headers: {
              "Content-Type":
                "application/x-www-form-urlencoded; charset=UTF-8",
              "User-Agent": utils.getRandomAndroidUA(),
              Referer: url,
            },
            data: postData,
          });
          if (!postResp.status) {
            return;
          }
          let postRespData = postResp.data;
          log.info(postRespData);
          let json_data = utils.toJSON(postRespData.responseText);
          let zt = json_data["zt"];
          let info = json_data["info"];
          if (zt === 4) {
            Qmsg.error(info);
          } else if (zt === 1) {
            let QmsgLoading = Qmsg.loading("获取文件夹成功,解析文件直链中...");
            /* 获取多文件的数组信息 */
            let folder = json_data["text"];
            /**
             * 弹出内容
             * @type {PopsFolderDataConfig[]}
             */
            let folderList = [];
            log.info(`本链接一共${folder.length}个文件`);
            for (let index = 0; index < folder.length; index++) {
              let folderInfo = folder[index];
              let fileShareCode = folderInfo["id"];
              let fileName = folderInfo["name_all"];
              let fileSize = folderInfo["size"];
              let fileType = folderInfo["icon"];
              let uploadTime = folderInfo["time"];
              folderList.push({
                fileName: fileName,
                fileSize: fileSize,
                fileType: fileType,
                createTime: uploadTime,
                latestTime: uploadTime,
                isFolder: false,
                index: 0,
                async clickEvent() {
                  let folderDownloadInfo = await that.parseMoreFile(
                    fileShareCode,
                    fileName,
                    fileSize,
                    uploadTime
                  );
                  /* 成功获取下载信息 */
                  if (folderDownloadInfo.success) {
                    return {
                      autoDownload: true,
                      mode: "aBlank",
                      url: folderDownloadInfo.downloadUrl,
                    };
                  } else {
                    log.error(["获取下载信息失败:", folderDownloadInfo]);
                    Qmsg.error(folderDownloadInfo.msg);
                  }
                },
              });
            }
            QmsgLoading.close();
            NetDiskUI.staticView.moreFile("蓝奏云文件解析", folderList);
          } else if ("密码不正确".indexOf(info) !== -1) {
            Qmsg.error("密码不正确!");
            NetDiskUI.newAccessCodeView(
              void 0,
              "lanzou",
              that.netDiskIndex,
              that.shareCode,
              (userInputAccessCode) => {
                that.init(
                  that.netDiskIndex,
                  that.shareCode,
                  userInputAccessCode
                );
              }
            );
          } else if ("没有了".indexOf(info) !== -1) {
            Qmsg.error("没有文件了");
          } else {
            Qmsg.error("未知错误");
          }
        };
        /**
         * 文件解析并返回html-vip
         * @param {string} paramShareCode 解析多文件获取的shareCode
         * @param {string} fileName 文件名
         * @param {string} fileSize 文件大小
         * @param {string} fileUploadTime 文件上传时间
         * @returns {Promise<{
         * success :boolean,
         * fileName: string,
         * fileSize: string,
         * fileUploadTime: string,
         * downloadUrl: ?string,
         * msg: string,
         * }>}
         */
        this.parseMoreFile = async function (
          paramShareCode,
          fileName,
          fileSize,
          fileUploadTime
        ) {
          /* 根据获取到的json中多文件链接来获取单文件直链 */
          let getResp = await httpx.get({
            url: that.router.default(paramShareCode),
            headers: {
              Accept: "*/*",
              "User-Agent": utils.getRandomPCUA(),
              Referer: that.router.default(that.shareCode),
            },
          });
          log.info(getResp);
          if (!getResp.status) {
            return {
              success: false,
              fileName: fileName,
              fileSize: fileSize,
              fileUploadTime: fileUploadTime,
              msg: `解析失败,${getResp.msg}`,
              downloadUrl: void 0,
            };
          }
          let respData = getResp.data;
          let pageText = respData.responseText;
          let pageDOM = DOMUtils.parseHTML(pageText, true, true);
          let pageIframeElement =
            pageDOM.querySelector('iframe[class^="ifr"]') ||
            pageDOM.querySelector('iframe[class^="n_downlink"]');
          if (!pageIframeElement) {
            return {
              success: false,
              fileName: fileName,
              fileSize: fileSize,
              fileUploadTime: fileUploadTime,
              msg: `解析iframe链接失败`,
              downloadUrl: void 0,
            };
          }
          let iframeUrl = pageIframeElement.getAttribute("src");
          log.error(["该链接需要重新通过iframe地址访问获取信息", iframeUrl]);
          Qmsg.info("正在请求下载信息");
          let downloadUrl = await that.getLinkByIframe(iframeUrl, {
            fileName,
            fileSize,
            fileUploadTime,
          });
          if (downloadUrl) {
            if (NetDiskFilterScheme.isForwardDownloadLink("lanzou")) {
              downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                "lanzou",
                downloadUrl
              );
            }
            return {
              success: true,
              fileName: fileName,
              fileSize: fileSize,
              fileUploadTime: fileUploadTime,
              msg: "success",
              downloadUrl: downloadUrl,
            };
          } else {
            return {
              success: false,
              fileName: fileName,
              fileSize: fileSize,
              fileUploadTime: fileUploadTime,
              msg: `获取下载链接失败`,
              downloadUrl: void 0,
            };
          }
        };
        return this;
      },
      /**
       * 蓝奏云优享
       * @constructor
       */
      lanzouyx: function () {
        let that = this;
        /** 所在规则的下标 */
        this.netDiskIndex = 0;
        /** 分享码 */
        this.shareCode = "";
        /** 分享码的id */
        this.shareCodeId = "";
        /** 提取码 */
        this.accessCode = "";
        /**
         * 获取的uuid
         * @type {?string}
         */
        this.uuid = void 0;
        /**
         * 获取的userId
         * @type {?string}
         **/
        this.userId = void 0;
        const LanZouUtils = {
          LanZouDiskApp: "lanZouY-disk-app",
          EncryptList: [
            "Y",
            "y",
            "0",
            "Z",
            "z",
            "N",
            "n",
            "M",
            "I",
            "6",
            "m",
            "W",
            "w",
            "1",
            "X",
            "x",
            "L",
            "l",
            "K",
            "7",
            "k",
            "i",
            "U",
            "u",
            "2",
            "V",
            "v",
            "J",
            "j",
            "8",
            "G",
            "g",
            "F",
            "S",
            "s",
            "3",
            "T",
            "t",
            "H",
            "h",
            "f",
            "E",
            "e",
            "D",
            "Q",
            "q",
            "4",
            "R",
            "r",
            "9",
            "d",
            "a",
            "C",
            "c",
            "B",
            "O",
            "o",
            "5",
            "P",
            "p",
            "b",
            "A",
          ],
          decodeChar(e) {
            for (let t = 0; t < this.EncryptList.length; t++)
              if (e == this.EncryptList[t]) return t;
            return -1;
          },
          /**
           * shareCode转id
           * @param {string} shareCode
           */
          idEncrypt(shareCode) {
            let t = 1,
              n = 0;
            if ("" != shareCode && shareCode.length > 4) {
              let r;
              shareCode = shareCode.substring(3, shareCode.length - 1);
              for (let index = 0; index < shareCode.length; index++)
                (r = shareCode.charAt(shareCode.length - index - 1)),
                  (n += this.decodeChar(r) * t),
                  (t *= 62);
            }
            return n;
          },
          encrypt(e) {
            const t = Cryptojs.enc.Utf8.parse(this.LanZouDiskApp),
              n = Cryptojs.enc.Utf8.parse(e),
              r = Cryptojs.AES.encrypt(n, t, {
                mode: Cryptojs.mode.ECB,
                padding: Cryptojs.pad.Pkcs7,
              });
            return r;
          },
          /**
           * 用于时间戳转加密字符串
           * @param {any} e
           * @returns
           */
          encryptHex(e) {
            const t = this.encrypt(e, this.LanZouDiskApp);
            return t.ciphertext.toString().toUpperCase();
          },
        };
        /**
         * 入口
         * @param {number} netDiskIndex
         * @param {string} shareCode
         * @param {string} accessCode
         */
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          that.shareCodeId = that.getDecodeShareCodeId(shareCode);
          that.uuid = that.getEncodeUUID();
          let linkInfo = await this.recommendList(
            3,
            "Chrome",
            that.uuid,
            2,
            that.getEncodeTimeStamp(),
            that.shareCodeId,
            0,
            1,
            60
          );
          if (!linkInfo) {
            return;
          }
          if (!linkInfo["list"].length) {
            return;
          }
          if (linkInfo["list"][0]?.["map"]?.["userId"]) {
            that.userId = linkInfo["list"][0]?.["map"]?.["userId"];
          } else {
            Qmsg.error("解析获取【userId】为空");
            return;
          }
          if (linkInfo["list"][0]["folderIds"] == null) {
            /* 单文件 */
            log.success("该链接是 单文件");
            let fileInfo = linkInfo["list"][0]["fileList"][0];
            let folderInfoList = that.parseFolderInfo(
              linkInfo["list"][0]["fileList"],
              0
            );
            /* 获取文件下载信息 */
            let downloadInfo = await folderInfoList[0]["clickEvent"]();
            if (downloadInfo) {
              let downloadUrl = downloadInfo["url"];
              if (NetDiskFilterScheme.isForwardDownloadLink("lanzouyx")) {
                downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                  "lanzouyx",
                  downloadUrl
                );
              }
              NetDiskUI.staticView.oneFile({
                title: "蓝奏云优享单文件直链",
                fileName: fileInfo["fileName"],
                fileSize: fileInfo["fileSize"] * 1024,
                downloadUrl: downloadUrl,
                fileUploadTime: utils.formatToTimeStamp(fileInfo["updTime"]),
                fileLatestTime: utils.formatToTimeStamp(fileInfo["updTime"]),
              });
            }
          } else {
            /* 多文件 */
            log.success("该链接是 多文件");
            Qmsg.info("正在递归文件");
            let QmsgLoading = Qmsg.loading(`正在解析多文件中,请稍后...`);
            let folderInfoList = that.parseFolderInfo(
              linkInfo["list"][0]["fileList"],
              0
            );
            QmsgLoading.close();
            log.info("递归完毕");
            NetDiskUI.staticView.moreFile("蓝奏云优享解析", folderInfoList);
          }
        };
        /**
         * 获取直链弹窗的文件夹信息
         * @param {object} infoList
         * @param {number} index
         */
        this.parseFolderInfo = function (infoList, index) {
          let folderInfoList = [];
          let tempFolderInfoList = [];
          /**
           * @type {PopsFolderDataConfig[]}
           */
          let tempFolderFileInfoList = [];
          infoList.forEach((item) => {
            if (item["fileType"] === 2) {
              /* 文件夹 */
              tempFolderInfoList.push({
                fileName: item["folderName"],
                fileSize: 0,
                fileType: "",
                createTime: new Date(item["updTime"]).getTime(),
                latestTime: new Date(item["updTime"]).getTime(),
                isFolder: true,
                index: index,
                async clickEvent() {
                  let nowTime = Date.now();
                  let timestamp = that.getEncodeTimeStamp(nowTime);
                  let folderId = item["folderId"];
                  let folderInfo = await that.getFolderInfo(
                    3,
                    "Chrome",
                    that.uuid,
                    2,
                    timestamp,
                    that.shareCodeId,
                    folderId,
                    1,
                    60
                  );
                  if (folderInfo && folderInfo["list"]) {
                    return that.parseFolderInfo(folderInfo["list"], index + 1);
                  } else {
                    return [];
                  }
                },
              });
            } else {
              /* 文件 */
              tempFolderFileInfoList.push({
                fileName: item["fileName"],
                fileSize: item["fileSize"] * 1024,
                fileType: "",
                createTime: new Date(item["updTime"]).getTime(),
                latestTime: new Date(item["updTime"]).getTime(),
                isFolder: false,
                index: index,
                async clickEvent() {
                  let fileId = item["fileId"];
                  let userId = item["userId"] || that.userId;
                  let uuid = that.uuid;
                  if (utils.isNull(userId)) {
                    Qmsg.error("获取【userId】为空");
                    return;
                  }
                  if (utils.isNull(uuid)) {
                    Qmsg.error("获取【uuid】为空");
                    return;
                  }
                  let downloadUrl = await that.getDownloadFileUrl(
                    ...that.getDownloadFileParams(fileId, userId, uuid)
                  );
                  if (downloadUrl) {
                    if (NetDiskFilterScheme.isForwardDownloadLink("lanzouyx")) {
                      downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                        "lanzouyx",
                        downloadUrl
                      );
                    }
                    return {
                      url: downloadUrl,
                      autoDownload: true,
                      mode: "aBlank",
                    };
                  }
                },
              });
            }
          });
          tempFolderInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          tempFolderFileInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          folderInfoList = folderInfoList.concat(tempFolderInfoList);
          folderInfoList = folderInfoList.concat(tempFolderFileInfoList);
          return folderInfoList;
        };
        /**
         * 获取列表信息
         * @param {number} devType
         * @param {string} devModel
         * @param {string} uuid
         * @param {number} extra
         * @param {string} timestamp
         * @param {number|string} shareId
         * @param {number} type
         * @param {number} offset
         * @param {number} limit
         * @returns
         */
        this.recommendList = async function (
          devType = 3,
          devModel = "Chrome",
          uuid = "",
          extra = 2,
          timestamp = "",
          shareId = "",
          type = 0,
          offset = 1,
          limit = 60
        ) {
          let postResp = await httpx.post(
            `https://api.ilanzou.com/unproved/recommend/list?devType=${devType}&devModel=${devModel}&uuid=${uuid}&extra=${extra}&timestamp=${timestamp}&shareId=${shareId}&type=${type}&offset=${offset}&limit=${limit}`,
            {
              headers: {
                Accept: "application/json, text/plain, */*",
                Origin: "https://www.ilanzou.com",
                Referer: "https://www.ilanzou.com/",
                "Sec-Fetch-Site": "same-site",
                Host: "api.ilanzou.com",
                "User-Agent": utils.getRandomPCUA(),
              },
              responseType: "json",
            }
          );
          if (!postResp.status) {
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          log.success(["获取链接信息:", data]);
          if (data["code"] !== 200) {
            Qmsg.error("请求链接信息失败");
            return;
          }
          if (!data["list"].length) {
            Qmsg.error("获取链接信息为空");
            return;
          }
          return data;
        };
        /**
         * 获取文件夹信息
         * @param {number} devType
         * @param {string} devModel
         * @param {string} uuid
         * @param {number} extra
         * @param {string} timestamp
         * @param {number|string} shareId
         * @param {number|string} folderId
         * @param {number} offset
         * @param {number} limit
         */
        this.getFolderInfo = async function (
          devType = 6,
          devModel = "Chrome",
          uuid = "",
          extra = 2,
          timestamp = "",
          shareId = "",
          folderId = "",
          offset = 1,
          limit = 60
        ) {
          let postResp = await httpx.post(
            `https://api.ilanzou.com/unproved/share/list?devType=${devType}&devModel=${devModel}&uuid=${uuid}&extra=${extra}&timestamp=${timestamp}&shareId=${shareId}&folderId=${folderId}&offset=${offset}&limit=${limit}`,
            {
              headers: {
                Accept: "application/json, text/plain, */*",
                Origin: "https://www.ilanzou.com",
                Referer: "https://www.ilanzou.com/",
                "Sec-Fetch-Site": "same-site",
                Host: "api.ilanzou.com",
                "User-Agent": utils.getRandomPCUA(),
              },
            }
          );
          if (!postResp.status) {
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          log.success(["获取文件列表信息:", data]);
          if (data["code"] === 200) {
            return data;
          } else {
            Qmsg.error(data["msg"]);
          }
        };
        /**
         * 获取下载链接
         */
        this.getDownloadFileUrl = async function (
          downloadId = "",
          enable = 1,
          devType = 6,
          uuid = "",
          timestamp = "",
          auth = ""
        ) {
          let getResp = await httpx.options(
            `https://api.ilanzou.com/unproved/file/redirect?downloadId=${downloadId}&enable=${enable}&devType=${devType}&uuid=${uuid}&timestamp=${timestamp}&auth=${auth}`
          );
          if (!getResp.status) {
            return;
          }
          log.success(getResp);
          if (getResp.data.responseText) {
            let errorData = utils.toJSON(getResp.data.responseText);
            log.error(errorData);
            Qmsg.error(errorData["msg"]);
            return;
          }
          return getResp.data.finalUrl;
        };
        /**
         * 获取加密的uuid
         * @param {number} e
         * @returns
         */
        this.getEncodeUUID = function (e = 21) {
          let r = (e = 21) =>
            crypto
              .getRandomValues(new Uint8Array(e))
              .reduce(
                (e, t) => (
                  (t &= 63),
                  (e +=
                    t < 36
                      ? t.toString(36)
                      : t < 62
                      ? (t - 26).toString(36).toUpperCase()
                      : t > 62
                      ? "-"
                      : "_"),
                  e
                ),
                ""
              );
          return r(e);
        };
        /**
         * 获取shareCode转换后的id
         */
        this.getDecodeShareCodeId = function (shareCode) {
          return LanZouUtils.idEncrypt(shareCode);
        };
        /**
         * 获取加密后的timestamp
         * @param {number} time
         */
        this.getEncodeTimeStamp = function (time = Date.now()) {
          return LanZouUtils.encryptHex(time);
        };
        /**
         * 获取下载文件的参数
         * @param {string} fileId 文件id
         * @param {string} userId 用户id
         * @param {?string} uuid 用户登录生成的uuid
         */
        this.getDownloadFileParams = function (fileId, userId = "", uuid) {
          let nowTime = Date.now();
          let downloadId = LanZouUtils.encryptHex(fileId + "|" + userId),
            enable = 1,
            devType = 6,
            timestamp = that.getEncodeTimeStamp(nowTime),
            auth = LanZouUtils.encryptHex(fileId + "|" + nowTime);
          return [downloadId, enable, devType, uuid, timestamp, auth];
        };
        /**
         * 前往登录
         */
        this.gotoLogin = function () {
          NetDiskPops.confirm(
            {
              title: {
                position: "center",
                text: "蓝奏云优享",
              },
              content: {
                text: "必须先在【蓝奏云优享】进行登录,然后登录成功后刷新获取必备的下载参数【uuid】和【userId】。",
                html: false,
              },
              btn: {
                reverse: true,
                position: "end",
                ok: {
                  text: "前往",
                  enable: true,
                  callback() {
                    window.open("https://www.ilanzou.com", "_blank");
                  },
                },
              },
            },
            NetDiskUI.popsStyle.tianYiYunLoginTip
          );
        };
        return this;
      },
      /**
       * 天翼云
       * + 开发文档:https://id.dlife.cn/html/api_detail_696.html
       * @constructor
       * @returns {object}
       */
      tianyiyun: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        let shareId = void 0;
        /* 猜测1是有密码,2是无密码 */
        let shareMode = 1;
        this.code = {
          ShareNotFoundFlatDir: "抱歉,该文件的分享平铺目录未找到",
          ShareNotFound: "抱歉,您访问的页面地址有误,或者该页面不存在。",
          ShareAuditNotPass: "抱歉,该内容审核不通过",
          FileNotFound: "抱歉,文件不存在",
          ShareExpiredError: "抱歉,您访问的页面地址有误,或者该页面不存在",
          ShareAuditWaiting: "抱歉,该链接处于审核中",
          ShareInfoNotFound: "抱歉,您访问的页面地址有误,或者该页面不存在",
          FileTooLarge: "抱歉,文件太大,不支持下载",
          InvalidSessionKey:
            "天翼云PC端Cookie未生成,是否前去登录?<br />&nbsp;&nbsp;&nbsp;&nbsp;(注意,需要当前浏览器的UA切换成PC且在登录后要等待进入个人云空间后生成Cookie,不是手机端浏览的个人云空间,那样生成的Cookie无法使用)",
        };
        /**
         *
         * @param {number} netDiskIndex
         * @param {string} shareCode
         * @param {string} accessCode
         * @returns
         */
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;

          let shareInfoData = await that.getShareInfoByCodeV2(shareCode);
          if (!shareInfoData) {
            return;
          }

          log.info(["解析的JSON信息", shareInfoData]);
          if (
            shareInfoData["needAccessCode"] &&
            utils.isNull(that.accessCode)
          ) {
            Qmsg.error("密码不正确!");
            NetDiskUI.newAccessCodeView(
              void 0,
              "tianyiyun",
              that.netDiskIndex,
              that.shareCode,
              (userInputAccessCode) => {
                that.init(
                  that.netDiskIndex,
                  that.shareCode,
                  userInputAccessCode
                );
              }
            );
            return;
          }
          if ("shareId" in shareInfoData) {
            shareId = shareInfoData["shareId"];
          } else {
            shareId = await that.getShareId(shareCode, accessCode);
          }
          if ("shareMode" in shareInfoData) {
            shareMode = shareInfoData["shareMode"];
          }
          if (shareId == void 0) {
            return;
          }
          if (shareInfoData.isFolder) {
            /* 多文件 */
            Qmsg.info("正在递归文件");
            let QmsgLoading = Qmsg.loading(`正在解析多文件中,请稍后...`);
            let fileId = shareInfoData["fileId"];
            let folderInfo = await that.listShareDir(
              shareCode,
              accessCode,
              void 0,
              void 0,
              fileId,
              fileId,
              void 0,
              shareId,
              void 0,
              void 0,
              void 0,
              void 0
            );
            if (!folderInfo) {
              QmsgLoading.close();
              return;
            }
            let folderInfoList = that.getFolderInfo(
              shareCode,
              accessCode,
              folderInfo,
              0
            );
            QmsgLoading.close();
            log.info("递归完毕");
            NetDiskUI.staticView.moreFile("天翼云文件解析", folderInfoList);
            return;
          } else {
            /* 单文件 */
            let downloadUrl = await that.getDownloadUrl(
              that.shareCode,
              that.accessCode,
              shareInfoData.fileId,
              shareId
            );
            if (downloadUrl) {
              if (NetDiskFilterScheme.isForwardDownloadLink("tianyiyun")) {
                downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                  "tianyiyun",
                  downloadUrl
                );
              }
              NetDiskUI.staticView.oneFile({
                title: "天翼云单文件直链",
                fileName: shareInfoData.fileName,
                fileSize: utils.formatByteToSize(shareInfoData.fileSize),
                downloadUrl: downloadUrl,
                fileUploadTime: shareInfoData.fileCreateDate,
                fileLatestTime: shareInfoData.fileLastOpTime,
              });
            }
          }
        };
        /**
         * 获取当前登录用户的信息
         * @returns {Promise<?{
         * encryptAccount: string,
         * icon: string,
         * nickname: string,
         * res_code: string,
         * res_message: string,
         * sessionKey: string,
         * userAccount: string
         * }>}
         */
        this.getUserBriefInfo = async function (shareCode) {
          let getResp = await httpx.get(
            "https://cloud.189.cn/api/portal/v2/getUserBriefInfo.action",
            {
              headers: {
                Accept: "application/json;charset=UTF-8",
                Referer: "https://cloud.189.cn/web/share?code=" + shareCode,
                "User-Agent": utils.getRandomPCUA(),
              },
              onerror() {},
            }
          );
          log.info(getResp);
          if (!getResp.status) {
            let errorResultJSON = utils.toJSON(getResp.data.responseText);
            if (errorResultJSON["res_code"] in that.code) {
              Qmsg.error(that.code[errorResultJSON["res_code"]]);
            } else {
              Qmsg.error("请求异常");
            }
            return;
          }
          let data = utils.toJSON(getResp.data.responseText);
          if (data["res_code"] === 0) {
            return data;
          }
        };
        /**
         * 获取分享信息
         * @param {string} shareCode
         * @returns
         */
        this.getShareInfoByCodeV2 = async function (shareCode) {
          let postResp = await httpx.post({
            url: "https://cloud.189.cn/api/open/share/getShareInfoByCodeV2.action",
            data: `shareCode=${shareCode}`,
            headers: {
              Accept: "application/json;charset=UTF-8",
              "Content-Type": "application/x-www-form-urlencoded",
              "User-Agent": utils.getRandomPCUA(),
              "Sign-Type": 1,
              Referer: "https://cloud.189.cn/web/share?code=" + shareCode,
              Origin: "https://cloud.189.cn",
            },
            onerror() {},
          });
          if (!postResp.status) {
            let errorData = utils.toJSON(postResp.data.responseText);
            log.error(["获取下载参数失败的JSON信息", errorData]);
            if (errorData["res_code"] in that.code) {
              Qmsg.error(that.code[errorData["res_code"]]);
            } else {
              Qmsg.error(errorData["res_message"]);
            }
            return;
          }
          let postData = postResp.data;
          log.info(postData);
          let data = utils.toJSON(postData.responseText);
          if (data["res_code"] == 0) {
            return data;
          } else {
            if (that.code.hasOwnProperty(data["res_code"])) {
              Qmsg.error(that.code[data["res_code"]]);
            } else {
              Qmsg.error("获取FileId失败");
            }
          }
        };
        /**
         * 获取shareId
         * @returns {Promise<?number>}
         */
        this.getShareId = async function (shareCode, accessCode) {
          let getResp = await httpx.get({
            url: `https://cloud.189.cn/api/open/share/checkAccessCode.action?shareCode=${shareCode}&accessCode=${accessCode}`,
            headers: {
              Accept: "application/json;charset=UTF-8",
              "Cache-Control": "no-cache",
              "User-Agent": utils.getRandomPCUA(),
              "Sign-Type": 1,
              Referer: `https://cloud.189.cn/web/share?code=${shareCode}`,
            },
            responseType: "json",
          });
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          log.info(respData);
          let data = utils.toJSON(respData.responseText);
          if (data["res_code"] === 0 && "shareId" in data) {
            return data["shareId"];
          } else {
            Qmsg.error("获取shareId失败");
            log.info(data);
          }
        };
        /**
         * 获取随机noCache
         * @returns {string}
         */
        this.getNoCacheValue = function () {
          let result = "";
          for (let index = 0; index < 17; index++) {
            result += utils.getRandomValue(1, 9);
          }
          return "0." + result;
        };
        /**
         * 获取下载链接
         * @param {string} shareCode
         * @param {string} accessCode
         * @param {number} fileId
         * @param {number} shareId
         * @returns {Promise}
         */
        this.getDownloadUrl = async function (
          shareCode,
          accessCode,
          fileId,
          shareId
        ) {
          let getResp = await httpx.get({
            url: `https://cloud.189.cn/api/open/file/getFileDownloadUrl.action?fileId=${fileId}&dt=1&shareId=${shareId}`,
            headers: {
              Accept: "application/json;charset=UTF-8",
              "Cache-Control": "no-cache",
              "User-Agent": utils.getRandomPCUA(),
              Referer: `https://cloud.189.cn/web/share?code=${shareCode}`,
              "Sign-Type": 1,
            },
            responseType: "json",
            onerror() {},
          });
          log.info(getResp);
          if (!getResp.status) {
            let errorResultJSON = utils.toJSON(getResp.data.responseText);
            if (errorResultJSON["errorCode"] === "InvalidSessionKey") {
              that.gotoLogin(that.code["InvalidSessionKey"]);
            } else if (errorResultJSON["res_code"] in that.code) {
              Qmsg.error(that.code[errorResultJSON["res_code"]]);
            } else {
              Qmsg.error("请求异常");
            }
            return;
          }
          let respData = getResp.data;
          let data = utils.toJSON(respData.responseText);
          log.info(data);
          if (data["res_code"] === 0) {
            return data["fileDownloadUrl"];
          } else if (
            "InvalidSessionKey" === data["res_code"] ||
            "InvalidSessionKey" === data["errorCode"]
          ) {
            that.gotoLogin(that.code["InvalidSessionKey"]);
          } else if (that.code.hasOwnProperty(data["res_code"])) {
            Qmsg.error(that.code[data["res_code"]]);
          } else {
            Qmsg.error("请求失败");
            log.error(respData);
          }
        };

        /**
         * 天翼云登录弹窗
         * @param {string} text 弹窗的显示的内容
         */
        this.gotoLogin = function (text = "") {
          NetDiskPops.confirm(
            {
              title: {
                position: "center",
                text: "天翼云",
              },
              content: {
                text: text,
                html: false,
              },
              btn: {
                reverse: true,
                position: "end",
                ok: {
                  text: "前往",
                  enable: true,
                  callback() {
                    window.open(
                      "https://cloud.189.cn/api/portal/loginUrl.action?redirectURL=https://cloud.189.cn/web/main",
                      "_blank"
                    );
                  },
                },
              },
            },
            NetDiskUI.popsStyle.tianYiYunLoginTip
          );
        };
        /**
         * 解析文件夹信息
         */
        this.listShareDir = async function (
          shareCode,
          accessCode,
          pageNum = 1,
          pageSize = 60,
          fileId,
          shareDirFileId,
          isFolder = true,
          shareId,
          iconOption = 5,
          orderBy = "lastOpTime",
          descending = true
        ) {
          /* Sessionkey: Sessionkey */
          let getResp = await httpx.get(
            `https://cloud.189.cn/api/open/share/listShareDir.action?pageNum=${pageNum}&pageSize=${pageSize}&fileId=${fileId}&shareDirFileId=${shareDirFileId}&isFolder=${isFolder}&shareId=${shareId}&shareMode=${shareMode}&iconOption=${iconOption}&orderBy=${orderBy}&descending=${descending}&accessCode=${accessCode}`,
            {
              headers: {
                Accept: "application/json;charset=UTF-8",
                Referer: `https://cloud.189.cn/web/share?code=${shareCode}`,
                "Sign-Type": 1,
                "User-Agent": utils.getRandomPCUA(),
              },
              responseType: "json",
              onerror() {},
            }
          );
          if (!getResp.status) {
            let errorData = utils.toJSON(getResp.data.responseText);
            log.error(["解析文件夹信息失败", errorData]);
            if (errorData["res_code"] in that.code) {
              Qmsg.error(that.code[errorData["res_code"]]);
            } else if ("res_message" in errorData) {
              Qmsg.error(errorData["res_message"]);
            } else {
              Qmsg.error("解析文件夹信息失败");
            }
            return;
          }
          let getData = getResp.data;
          log.info(getData);
          let data = utils.toJSON(getData.responseText);
          if (data["res_code"] == 0) {
            return data["fileListAO"];
          } else {
            if (that.code.hasOwnProperty(data["res_code"])) {
              Qmsg.error(that.code[data["res_code"]]);
            } else {
              Qmsg.error("获取FileId失败");
            }
          }
        };
        /**
         * 获取直链弹窗的文件夹信息
         */
        this.getFolderInfo = function (
          shareCode,
          accessCode,
          dirInfo,
          index = 0
        ) {
          /**
           * @type {PopsFolderDataConfig[]}
           */
          let folderInfoList = [];
          let tempFolderInfoList = [];
          let tempFolderFileInfoList = [];
          /* 文件夹 */
          dirInfo["folderList"].forEach((folderInfo) => {
            folderInfoList.push({
              fileName: folderInfo["name"],
              fileSize: 0,
              fileType: "",
              createTime: utils.formatToTimeStamp(folderInfo["createDate"]),
              latestTime: utils.formatToTimeStamp(folderInfo["lastOpTime"]),
              isFolder: true,
              index: index,
              async clickEvent() {
                let _folderInfo_ = await that.listShareDir(
                  shareCode,
                  accessCode,
                  1,
                  100,
                  folderInfo["id"],
                  folderInfo["id"],
                  void 0,
                  shareId,
                  void 0,
                  void 0,
                  void 0,
                  void 0
                );
                if (!_folderInfo_) {
                  return [];
                }
                return that.getFolderInfo(
                  shareCode,
                  accessCode,
                  _folderInfo_,
                  index + 1
                );
              },
            });
          });
          /* 文件 */
          dirInfo["fileList"].forEach((fileInfo) => {
            folderInfoList.push({
              fileName: fileInfo["name"],
              fileSize: fileInfo["size"],
              fileType: "",
              createTime: utils.formatToTimeStamp(fileInfo["createDate"]),
              latestTime: utils.formatToTimeStamp(fileInfo["lastOpTime"]),
              isFolder: false,
              index: index,
              async clickEvent() {
                let downloadUrl = await that.getDownloadUrl(
                  shareCode,
                  accessCode,
                  fileInfo["id"],
                  shareId
                );
                if (downloadUrl) {
                  if (NetDiskFilterScheme.isForwardDownloadLink("tianyiyun")) {
                    downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                      "tianyiyun",
                      downloadUrl
                    );
                  }

                  return {
                    autoDownload: true,
                    mode: "aBlank",
                    url: downloadUrl,
                  };
                }
              },
            });
          });
          tempFolderInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          tempFolderFileInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          folderInfoList = folderInfoList.concat(tempFolderInfoList);
          folderInfoList = folderInfoList.concat(tempFolderFileInfoList);
          log.info(["getFolderInfo", folderInfoList]);
          return folderInfoList;
        };
        return this;
      },
      /**
       * 文叔叔
       * @constructor
       * @returns {object}
       */
      wenshushu: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        /**
         * 用于header头x-token
         * @type {string}
         */
        let token = void 0;
        this.code = {
          1004: "no token",
          1008: "您没有权限访问",
          1013: "糟糕,此任务已过期销毁,下次要记得续期",
          1066: "对方设置的下载 / 预览次数已用完",
          1088: "糟糕,您访问的页面不存在",
        };
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          Qmsg.info("正在请求直链中...");
          let token = await this.getWssToken();
          if (!token) {
            return;
          }
          let pidInfo = await this.getPid();
          if (!pidInfo) {
            return;
          }
          await this.getFileNList(pidInfo.bid, pidInfo.pid);
        };
        /**
         * 获取token
         * wss:xxxxxx
         * @returns {Promise<string>}
         */
        this.getWssToken = async function () {
          let postResp = await httpx.post({
            url: "https://www.wenshushu.cn/ap/login/anonymous",
            responseType: "json",
            dataType: "json",
            data: JSON.stringify({
              dev_info: "{}",
            }),
            headers: {
              Accept: "application/json, text/plain, */*",
              "User-Agent": utils.getRandomAndroidUA(),
              Referer: "https://www.wenshushu.cn/f/" + that.shareCode,
            },
          });
          log.success(postResp);
          if (!postResp.status) {
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          if (data["code"] === 0) {
            token = data["data"]["token"];
            return data["data"]["token"];
          } else if (data["code"] in that.code) {
            Qmsg.error(that.code[data["code"]]);
          } else {
            Qmsg.error("获取wss失败");
          }
        };
        /**
         * 获取pid
         * @returns {Promise<{bid:string,pid:string}> }
         */
        this.getPid = async function () {
          let postResp = await httpx.post({
            url: "https://www.wenshushu.cn/ap/task/mgrtask",
            dataType: "json",
            responseType: "json",
            data: JSON.stringify({
              tid: that.shareCode,
              password: "",
              ufileid: "",
            }),
            headers: {
              Accept: "application/json, text/plain, */*",
              "User-Agent": utils.getRandomAndroidUA(),
              Referer: "https://www.wenshushu.cn/f/" + that.shareCode,
              "x-token": token,
            },
          });
          log.success(postResp);
          if (!postResp.status) {
            return;
          }
          let respData = postResp.data;
          let data = utils.toJSON(respData.responseText);
          if (data["code"] === 0) {
            return {
              bid: data["data"]["boxid"],
              pid: data["data"]["ufileid"],
            };
          } else if (data["code"] in that.code) {
            Qmsg.error(that.code[data["code"]]);
          } else {
            Qmsg.error("获取pid失败");
          }
        };
        /**
         * 获取文件列表信息
         * @param {string} bid
         * @param {string} pid
         * @returns
         */
        this.getFileNList = async function (bid, pid) {
          let postResp = await httpx.post({
            url: "https://www.wenshushu.cn/ap/ufile/nlist",
            dataType: "json",
            responseType: "json",
            data: JSON.stringify({
              start: 0,
              sort: {
                name: "asc",
              },
              bid: bid,
              pid: pid,
              options: {
                uploader: "true",
              },
              size: 50,
            }),
            headers: {
              Accept: "application/json, text/plain, */*",
              "User-Agent": utils.getRandomAndroidUA(),
              Referer: "https://www.wenshushu.cn/f/" + that.shareCode,
              "x-token": token,
            },
          });
          log.success(postResp);
          if (!postResp.status) {
            return;
          }
          let respData = postResp.data;
          let jsonData = utils.toJSON(respData.responseText);
          if (jsonData["code"] === 0) {
            if (jsonData["data"]["fileList"][0]["type"] === 2) {
              Qmsg.error("该链接为多层级文件嵌套,跳转");
              NetDiskParse.blank(
                NetDiskParse.getBlankUrl(
                  "wenshushu",
                  that.netDiskIndex,
                  that.shareCode,
                  that.accessCode
                ),
                "wenshushu",
                that.netDiskIndex,
                that.shareCode,
                that.accessCode
              );
            } else {
              await that.getDownloadUrl(jsonData["data"]["fileList"][0]);
            }
          } else if (jsonData["code"] in that.code) {
            Qmsg.error(that.code[jsonData["code"]]);
          } else {
            Qmsg.error("获取文件信息失败");
          }
        };
        /**
         * 获取下载链接
         * @param {object} data
         * @returns {Promise}
         */
        this.getDownloadUrl = async function (data) {
          let file_name = data.fname;
          let file_size = utils.formatByteToSize(data.size);
          let postResp = await httpx.post({
            url: "https://www.wenshushu.cn/ap/dl/sign",
            dataType: "json",
            responseType: "json",
            data: JSON.stringify({
              ufileid: data.fid,
              consumeCode: 0,
            }),
            headers: {
              Accept: "application/json, text/plain, */*",
              "User-Agent": utils.getRandomAndroidUA(),
              Referer: "https://www.wenshushu.cn/f/" + that.shareCode,
              "x-token": token,
            },
          });
          if (!postResp.status) {
            return;
          }
          log.success(postResp);
          let respData = postResp.data;
          let jsonData = utils.toJSON(respData.responseText);
          if (jsonData["code"] == 0) {
            let downloadUrl = jsonData["data"]["url"];
            if (downloadUrl === "") {
              Qmsg.error("对方的分享流量不足");
            } else {
              if (NetDiskFilterScheme.isForwardDownloadLink("wenshushu")) {
                downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                  "wenshushu",
                  downloadUrl
                );
              }

              /* 文叔叔没有上传时间信息(暂时是这样的) */
              NetDiskUI.staticView.oneFile({
                title: "文叔叔单文件直链",
                fileName: file_name,
                fileSize: file_size,
                downloadUrl: downloadUrl,
              });
            }
          } else if (jsonData["data"] in that.code) {
            Qmsg.error(that.code[jsonData["data"]]);
          } else {
            Qmsg.error("获取下载链接失败");
          }
        };
        return this;
      },
      /**
       * 123盘
       * @constructor
       * @returns {object}
       */
      _123pan: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        this.code = {
          5103: "分享码错误或者分享地址错误",
          5104: "分享已过期",
          "-1000": "获取出错",
          "-2000": "请求异常",
          "-3000": "请求意外中止",
          "-4000": "请求超时",
          104: "文件已失效",
        };
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          that.panelList = [];
          that.Authorization = GM_getValue("_123pan_User_Authorization");
          let checkLinkValidityStatus = await that.checkLinkValidity();
          if (!checkLinkValidityStatus) {
            return;
          }
          let infoLists = await that.getFiles();
          if (!infoLists) {
            return;
          }
          if (infoLists.length === 1 && infoLists[0]["Type"] == 0) {
            let fileInfo = infoLists[0];
            if (fileInfo["Status"] == 104) {
              Qmsg.error("文件已失效");
              return;
            }
            let downloadUrl = fileInfo["DownloadUrl"];
            let fileSize = "";
            if (downloadUrl === "") {
              let downloadInfo = await that.getFileDownloadInfo(
                fileInfo["Etag"],
                fileInfo["FileId"],
                fileInfo["S3KeyFlag"],
                that.shareCode,
                fileInfo["Size"]
              );
              if (downloadInfo && downloadInfo["code"] === 0) {
                if (NetDiskFilterScheme.isForwardDownloadLink("_123pan")) {
                  downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                    "_123pan",
                    downloadInfo["data"]["DownloadURL"]
                  );
                }
                fileSize = utils.formatByteToSize(fileInfo["Size"]);
              } else if (downloadInfo && downloadInfo["code"] === 401) {
                downloadUrl = "javascript:;";
                fileSize = "请登录后下载";
              } else {
                downloadUrl = "javascript:;";
                fileSize = "获取下载链接失败";
              }
            } else {
              if (NetDiskFilterScheme.isForwardDownloadLink("_123pan")) {
                downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                  "_123pan",
                  downloadUrl
                );
              }

              fileSize = utils.formatByteToSize(fileInfo["Size"]);
            }
            let fileUploadTime = new Date(fileInfo["CreateAt"]).getTime();
            let fileLatestTime = new Date(fileInfo["UpdateAt"]).getTime();
            fileUploadTime = utils.formatTime(fileUploadTime);
            fileLatestTime = utils.formatTime(fileLatestTime);
            NetDiskUI.staticView.oneFile({
              title: "123盘单文件直链",
              fileName: fileInfo["FileName"],
              fileSize: fileSize,
              downloadUrl: downloadUrl,
              fileUploadTime: fileUploadTime,
              fileLatestTime: fileLatestTime,
            });
          } else {
            Qmsg.info("正在递归文件");
            let QmsgLoading = Qmsg.loading(`正在解析多文件中,请稍后...`);
            let folderInfoList = that.getFolderInfo(infoLists, 0);
            QmsgLoading.close();
            log.info("递归完毕");
            NetDiskUI.staticView.moreFile("123盘文件解析", folderInfoList);
          }
        };
        /**
         * 校验链接有效性
         * @returns {boolean}
         */
        this.checkLinkValidity = async function () {
          Qmsg.info("正在校验链接有效性");
          let url = `https://www.123pan.com/s/${that.shareCode}`;

          let getResp = await httpx.get({
            url: url,
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Referer: "https://www.123pan.com",
            },
          });
          log.info(getResp);
          if (!getResp.status) {
            return false;
          }
          let respData = getResp.data;
          let g_initialPropsMatch = respData.responseText.match(
            /window.g_initialProps[\s]*=[\s]*\{(.+?)\};/s
          );
          if (g_initialPropsMatch) {
            log.info(g_initialPropsMatch);
            let g_initialProps = utils.toJSON(
              `{${g_initialPropsMatch[g_initialPropsMatch.length - 1]}}`
            );
            log.info(g_initialProps);
            if (g_initialProps.res.code !== 0) {
              Qmsg.error(g_initialProps.res.message);
              return false;
            }
            let HasPwd = g_initialProps.res.data.HasPwd;
            if (
              HasPwd &&
              (that.accessCode == void 0 || that.accessCode === "")
            ) {
              /* 该链接需要密码但是没有获取到 */
              Qmsg.error("密码缺失!");
              NetDiskUI.newAccessCodeView(
                "密码缺失",
                "_123pan",
                that.netDiskIndex,
                that.shareCode,
                (userInputAccessCode) => {
                  that.init(
                    that.netDiskIndex,
                    that.shareCode,
                    userInputAccessCode
                  );
                }
              );
            } else {
              /* 该链接不需要密码 || 该链接需要密码且已获取到 */
              return true;
            }
          } else {
            Qmsg.error("校验链接-获取初始化内容失败");
          }
        };
        /**
         * 获取文件
         * @param {number} parentFileId
         * @returns {Promise<?{
         * Category: number,
         * ContentType: string,
         * CreateAt: number,
         * DownloadUrl: string,
         * Etag: string,
         * FileId: number,
         * FileName: string,
         * ParentFileId: number,
         * PunishFlag: number,
         * S3KeyFlag: number,
         * Size: number,
         * Status: number,
         * StorageNode: string,
         * Type: 0|1,
         * UpdateAt: string,
         * }[]>}
         */
        this.getFiles = async function (parentFileId = 0) {
          let url = `https://www.123pan.com/b/api/share/get?limit=100&next=1&orderBy=share_id&orderDirection=desc&shareKey=${that.shareCode}&SharePwd=${that.accessCode}&ParentFileId=${parentFileId}&Page=1`;
          let getResp = await httpx.get({
            url: url,
            headers: {
              Accept: "*/*",
              "User-Agent": utils.getRandomPCUA(),
              Referer: `https://www.123pan.com/s/${that.shareCode}`,
            },
          });
          log.info(getResp);
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          let json_data = utils.toJSON(respData.responseText);
          if (json_data["code"] === 0) {
            let infoList = json_data["data"]["InfoList"];
            return infoList;
          } else if (json_data["code"] === 5103) {
            NetDiskUI.newAccessCodeView(
              void 0,
              "_123pan",
              that.netDiskIndex,
              that.shareCode,
              (userInputAccessCode) => {
                that.init(
                  that.netDiskIndex,
                  that.shareCode,
                  userInputAccessCode
                );
              }
            );
          } else if (that.code[json_data["code"]]) {
            Qmsg.error(that.code[json_data["code"]]);
          } else if ("message" in json_data) {
            Qmsg.error(json_data["message"]);
          } else {
            Qmsg.error("123盘:未知的JSON格式");
          }
        };
        /**
         * 递归算法使用的请求
         * @param {string} parentFileId
         * @returns {Promise<?{
         * Category: number,
         * ContentType: string,
         * CreateAt: number,
         * DownloadUrl: string,
         * Etag: string,
         * FileId: number,
         * FileName: string,
         * ParentFileId: number,
         * PunishFlag: number,
         * S3KeyFlag: number,
         * Size: number,
         * Status: number,
         * StorageNode: string,
         * Type: 0|1,
         * UpdateAt: string,
         * }[]>}
         */
        this.getFilesByRec = async function (parentFileId) {
          let getResp = await httpx.get({
            url: `https://www.123pan.com/b/api/share/get?limit=100&next=1&orderBy=share_id&orderDirection=desc&shareKey=${that.shareCode}&SharePwd=${that.accessCode}&ParentFileId=${parentFileId}&Page=1`,
            headers: {
              Accept: "*/*",
              "User-Agent": utils.getRandomAndroidUA(),
              Referer: `https://www.123pan.com/s/${that.shareCode}`,
            },
          });
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          log.info(respData);
          let jsonData = utils.toJSON(respData.responseText);
          if (jsonData["code"] == 0) {
            return jsonData["data"]["InfoList"];
          }
        };
        /**
         * 获取文件夹信息
         * @param {{
         * Category: number,
         * ContentType: string,
         * CreateAt: number,
         * DownloadUrl: string,
         * Etag: string,
         * FileId: number,
         * FileName: string,
         * ParentFileId: number,
         * PunishFlag: number,
         * S3KeyFlag: number,
         * Size: number,
         * Status: number,
         * StorageNode: string,
         * Type: 0|1,
         * UpdateAt: string,
         * }[]} infoList
         * @returns {Promise<{
         * fileName: string,
         * fileSize: string|number,
         * fileType: ?string,
         * createTime: ?string,
         * latestTime: ?string,
         * isFolder: boolean,
         * index: ?number,
         * clickCallBack: ?(event:Event,_config_: object)=>{}
         * }[]>}
         */
        this.getFolderInfo = function (infoList, index) {
          let folderInfoList = [];
          let tempFolderInfoList = [];
          /**
           * @type {PopsFolderDataConfig[]}
           */
          let tempFolderFileInfoList = [];
          infoList.forEach((item) => {
            if (item.Type) {
              /* 文件夹 */
              tempFolderInfoList.push({
                fileName: item.FileName,
                fileSize: 0,
                fileType: "",
                createTime: new Date(item.CreateAt).getTime(),
                latestTime: new Date(item.UpdateAt).getTime(),
                isFolder: true,
                index: index,
                async clickEvent() {
                  let resultFileInfoList = await that.getFilesByRec(
                    item["FileId"]
                  );
                  if (resultFileInfoList) {
                    return that.getFolderInfo(resultFileInfoList, index + 1);
                  } else {
                    return [];
                  }
                },
              });
            } else {
              /* 文件 */
              tempFolderFileInfoList.push({
                fileName: item.FileName,
                fileSize: item.Size,
                fileType: "",
                createTime: new Date(item.CreateAt).getTime(),
                latestTime: new Date(item.UpdateAt).getTime(),
                isFolder: false,
                index: index,
                async clickEvent() {
                  if (item.Status == 104) {
                    Qmsg.error("文件已失效");
                  } else if (!Boolean(item.DownloadUrl)) {
                    let downloadInfo = await that.getFileDownloadInfo(
                      item["Etag"],
                      item["FileId"],
                      item["S3KeyFlag"],
                      that.shareCode,
                      item["Size"]
                    );
                    if (downloadInfo && downloadInfo["code"] === 0) {
                      return {
                        url: downloadInfo["data"]["DownloadURL"],
                        autoDownload: true,
                        mode: "aBlank",
                      };
                    } else if (downloadInfo && downloadInfo["code"] === 401) {
                      Qmsg.error("请登录后下载");
                    } else {
                      Qmsg.error("获取下载链接失败");
                    }
                  } else {
                    let downloadUrl = item.DownloadUrl;
                    if (NetDiskFilterScheme.isForwardDownloadLink("_123pan")) {
                      downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                        "_123pan",
                        downloadUrl
                      );
                    }
                    return {
                      url: downloadUrl,
                      autoDownload: true,
                      mode: "aBlank",
                    };
                  }
                },
              });
            }
          });
          tempFolderInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          tempFolderFileInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          folderInfoList = folderInfoList.concat(tempFolderInfoList);
          folderInfoList = folderInfoList.concat(tempFolderFileInfoList);
          return folderInfoList;
        };
        /**
         * 获取单文件下载链接
         * 123云盘新增了下载验证
         * @param {string} Etag
         * @param {string} FileID
         * @param {string} S3keyFlag
         * @param {string} ShareKey
         * @param {string} Size
         * @returns
         */
        this.getFileDownloadInfo = async function (
          Etag,
          FileID,
          S3keyFlag,
          ShareKey,
          Size
        ) {
          let authK_V = that.getFileDownloadAuth();
          let headers = {
            "App-Version": "3",
            Platform: "web",
            "Content-Type": "application/json;charset=UTF-8",
            Host: "www.123pan.com",
            Accept: "*/*",
            "User-Agent": utils.getRandomPCUA(),
            Referer: "https://www.123pan.com/s/" + ShareKey,
            Origin: "https://www.123pan.com",
          };
          if (that.Authorization) {
            headers["Authorization"] = "Bearer " + that.Authorization;
          }
          log.success("获取下载链接加密参数:" + authK_V);
          let postResp = await httpx.post({
            url: `https://www.123pan.com/a/api/share/download/info?${authK_V[0]}=${authK_V[1]}`,
            data: JSON.stringify({
              Etag: Etag,
              FileID: FileID,
              S3keyFlag: S3keyFlag,
              ShareKey: ShareKey,
              Size: Size,
            }),
            responseType: "json",
            headers: headers,
          });
          if (!postResp.status) {
            return;
          }
          let postData = postResp.data;
          let jsonData = utils.toJSON(postData.responseText);
          log.info(jsonData);
          if (jsonData["code"] == 0) {
            jsonData["data"]["DownloadURL"] = await that.decodeDownloadUrl(
              jsonData["data"]["DownloadURL"]
            );
            return jsonData;
          } else {
            return {
              code: jsonData["code"],
            };
          }
        };
        /**
         * 获取单文件下载链接的加密参数
         * 感谢:https://github.com/qaiu/netdisk-fast-download/
         */
        this.getFileDownloadAuth = function () {
          function encry_time(param) {
            var param_time,
              param_other =
                arguments["length"] > 0x2 && void 0x0 !== arguments[0x2]
                  ? arguments[0x2]
                  : 0x8;
            if (0x0 === arguments["length"]) return void 0;
            "object" === typeof param
              ? (param_time = param)
              : (0xa === ("" + param)["length"] &&
                  (param = 0x3e8 * parseInt(param)),
                (param_time = new Date(param)));
            var param_timezoneoffset =
                param + 0xea60 * new Date(param)["getTimezoneOffset"](),
              param_time_n = param_timezoneoffset + 0x36ee80 * param_other;
            return (
              (param_time = new Date(param_time_n)),
              {
                y: param_time["getFullYear"](),
                m:
                  param_time["getMonth"]() + 0x1 < 0xa
                    ? "0" + (param_time["getMonth"]() + 0x1)
                    : param_time["getMonth"]() + 0x1,
                d:
                  param_time["getDate"]() < 0xa
                    ? "0" + param_time["getDate"]()
                    : param_time["getDate"](),
                h:
                  param_time["getHours"]() < 0xa
                    ? "0" + param_time["getHours"]()
                    : param_time["getHours"](),
                f:
                  param_time["getMinutes"]() < 0xa
                    ? "0" + param_time["getMinutes"]()
                    : param_time["getMinutes"](),
              }
            );
          }

          function encry_join(param) {
            for (
              var a =
                  arguments["length"] > 0x1 && void 0x0 !== arguments[0x1]
                    ? arguments[0x1]
                    : 0xa,
                funcRun = function () {
                  for (var b, c = [], d = 0x0; d < 0x100; d++) {
                    b = d;
                    for (var index = 0x0; index < 0x8; index++)
                      b = 0x1 & b ? 0xedb88320 ^ (b >>> 0x1) : b >>> 0x1;
                    c[d] = b;
                  }
                  return c;
                },
                _funcRun_ = funcRun(),
                _param = param,
                _param_1 = -0x1,
                _param_0 = 0x0;
              _param_0 < _param["length"];
              _param_0++
            )
              _param_1 =
                (_param_1 >>> 0x8) ^
                _funcRun_[0xff & (_param_1 ^ _param.charCodeAt(_param_0))];
            return (_param_1 = (-0x1 ^ _param_1) >>> 0x0), _param_1.toString(a);
          }

          function getSign(urlPath) {
            var param_web = "web";
            var param_type = 3;
            var param_time = Math.round(
              (new Date().getTime() +
                0x3c * new Date().getTimezoneOffset() * 0x3e8 +
                28800000) /
                0x3e8
            ).toString();
            var key = "a,d,e,f,g,h,l,m,y,i,j,n,o,p,k,q,r,s,t,u,b,c,v,w,s,z";
            var randomRoundNum = Math["round"](0x989680 * Math["random"]());

            var number_split;
            var time_a;
            var time_y;
            var time_m;
            var time_d;
            var time_h;
            var time_f;
            var time_array;
            var time_push;
            for (var number_item in ((number_split = key.split(",")),
            (time_a = encry_time(param_time)),
            (time_y = time_a["y"]),
            (time_m = time_a["m"]),
            (time_d = time_a["d"]),
            (time_h = time_a["h"]),
            (time_f = time_a["f"]),
            (time_array = [time_y, time_m, time_d, time_h, time_f].join("")),
            (time_push = []),
            time_array))
              time_push["push"](number_split[Number(time_array[number_item])]);
            var param_no;
            var param_join_s;
            return (
              (param_no = encry_join(time_push["join"](""))),
              (param_join_s = encry_join(
                ""
                  ["concat"](param_time, "|")
                  ["concat"](randomRoundNum, "|")
                  ["concat"](urlPath, "|")
                  ["concat"](param_web, "|")
                  ["concat"](param_type, "|")
                  ["concat"](param_no)
              )),
              [
                param_no,
                ""
                  ["concat"](param_time, "-")
                  ["concat"](randomRoundNum, "-")
                  ["concat"](param_join_s),
              ]
            );
          }
          return getSign("/a/api/share/download/info");
        };
        /**
         * 将直链的param参数解析成真正的直链
         * @param {string} url
         * @returns
         */
        this.decodeDownloadUrl = async function (url) {
          if (url === "") {
            return "";
          }
          let decodeURL = new URL(url);
          let params = decodeURL.search.replace(/^\?params=/gi, "");
          params = params.split("&")[0];
          try {
            let newDecodeUrl = decodeURI(atob(params));
            Qmsg.info("正在获取重定向直链");
            let getResp = await httpx.get({
              url: newDecodeUrl,
              responseType: "json",
              headers: {
                "User-Agent": utils.getRandomAndroidUA(),
                Referer: "https://www.123pan.com/s/" + that.shareCode,
                Origin: "https://www.123pan.com",
              },
              onerror: function () {},
            });
            log.info(getResp);
            if (!getResp.status && getResp.data.status !== 210) {
              /* 很奇怪,123盘返回的状态码是210 */
              return newDecodeUrl;
            }
            let respData = getResp.data;
            let resultJSON = utils.toJSON(respData.responseText);
            let newURL = new URL(resultJSON.data.redirect_url);
            newURL.searchParams.set("auto_redirect", 1);
            log.success(resultJSON);
            return newURL.toString();
          } catch (error) {
            log.error(error);
            return url;
          }
        };
        return this;
      },
      /**
       * 坚果云
       * @constructor
       * @returns {object}
       */
      jianguoyun: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        this.errorCode = {
          UnAuthorized: "请先登录坚果云账号",
        };
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          let downloadParams = await that.getRequestDownloadParams();
          if (!downloadParams) {
            return;
          }
          if (downloadParams["isdir"]) {
            /* 是文件夹 */
            let Qmsg_loading = Qmsg.loading("正在遍历多文件信息...");
            let folderInfo = await that.getFolderInfo(downloadParams["hash"]);
            if (!folderInfo) {
              Qmsg_loading.close();
              return;
            }
            let newFolderInfoList = that.parseMoreFile(
              folderInfo,
              downloadParams["hash"],
              downloadParams["name"]
            );
            Qmsg_loading.close();

            /* 坚果云盘没有上传时间信息(暂时是这样的) */
            NetDiskUI.staticView.moreFile("坚果云文件解析", newFolderInfoList);
          } else {
            /* 是文件 */
            let fileSize = utils.formatByteToSize(downloadParams["size"]);
            let downloadUrl = await that.getFileLink(
              downloadParams.hash,
              downloadParams.name
            );
            if (!downloadUrl) {
              return;
            }
            if (NetDiskFilterScheme.isForwardDownloadLink("jianguoyun")) {
              downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                "jianguoyun",
                downloadUrl
              );
            }

            log.info(downloadUrl);
            /* 坚果云盘没有上传时间信息(暂时是这样的) */
            NetDiskUI.staticView.oneFile({
              title: "坚果云盘单文件直链",
              fileName: downloadParams["name"],
              fileSize: fileSize,
              downloadUrl: downloadUrl,
            });
          }
        };
        /**
         * 解析多文件信息
         * @param {{
         * mtime: number,
         * relPath: string,
         * size: number,
         * tblUri: ?string,
         * type: "file"|string,
         * }[]} folderInfo
         * @param {string} hash 文件hash值
         * @param {string} fileName 文件名
         * @returns {{
         * fileName: string,
         * fileSize: string|number,
         * fileType: ?string,
         * createTime: ?string,
         * latestTime: ?string,
         * isFolder: boolean,
         * index: ?number,
         * clickCallBack: ?(event:Event,_config_: object)=>{}
         * }[]}
         */
        this.parseMoreFile = function (folderInfo, hash = "", fileName = "") {
          log.info(["解析多文件信息", folderInfo]);
          /**
           * @type {PopsFolderDataConfig[]}
           */
          let folderInfoList = [];
          folderInfo.forEach((item) => {
            let fileName = item.relPath;
            if (fileName.startsWith("/")) {
              fileName = fileName.replace(/^\//, "");
            }
            folderInfoList.push({
              fileName: fileName,
              fileSize: item["size"],
              fileType: "",
              createTime: item.mtime,
              latestTime: item.mtime,
              isFolder: false,
              index: 0,
              async clickEvent() {
                Qmsg.info("正在获取下载链接...");
                let downloadUrl = await that.getDirLink(
                  hash,
                  fileName,
                  item["relPath"]
                );
                if (!downloadUrl) {
                  return;
                }
                Qmsg.success("获取成功!");
                if (NetDiskFilterScheme.isForwardDownloadLink("jianguoyun")) {
                  downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                    "jianguoyun",
                    downloadUrl
                  );
                }

                log.info(downloadUrl);
                return {
                  autoDownload: true,
                  mode: "aBlank",
                  url: downloadUrl,
                };
              },
            });
          });
          return folderInfoList;
        };
        /**
         * 获取下载链接所需要的hash值和name
         */
        this.getRequestDownloadParams = async function () {
          log.info("获取hash值");
          Qmsg.info("正在获取请求信息");
          let pageInfoRegexp = /var[\s]*PageInfo[\s]*=[\s]*{([\s\S]+)};/i;
          let formData = new FormData();
          formData.append("pd", that.accessCode);
          let requestDetails = {
            url: `https://www.jianguoyun.com/p/${that.shareCode}`,
            data: that.accessCode === "" ? void 0 : `pd=${that.accessCode}`,
            responseType: "html",
            headers: {
              "Content-Type": "application/x-www-form-urlencoded",
              "User-Agent": utils.getRandomPCUA(),
              Referer: `https://www.jianguoyun.com/p/${that.shareCode}`,
            },
          };
          let requestResp = void 0;
          if (that.accessCode === "") {
            requestResp = await httpx.get(requestDetails);
          } else {
            requestResp = await httpx.post(requestDetails);
          }
          if (!requestResp.status) {
            return;
          }
          let respData = requestResp.data;
          log.info("请求信息");
          log.info(respData);
          let pageInfo = respData.responseText.match(pageInfoRegexp);
          if (pageInfo) {
            pageInfo = pageInfo[pageInfo.length - 1];
            pageInfo = `({${pageInfo}})`;
            pageInfo = eval(pageInfo);
            log.info(pageInfo);
            let fileName = pageInfo["name"];
            let fileSize = pageInfo["size"];
            let fileHash = pageInfo["hash"];
            let fileNeedsPassword = pageInfo["needsPassword"];
            let fileOwner = pageInfo["owner"];
            let isdir = pageInfo["isdir"];
            let fileErrorCode = pageInfo["errorCode"];
            fileName = decodeURIComponent(fileName);
            log.success("是否是文件夹 ===> " + isdir);
            log.success("hash ===> " + fileHash);
            log.success("name ===> " + fileName);
            log.success("size ===> " + fileSize);
            if (
              fileNeedsPassword &&
              (that.accessCode == void 0 || that.accessCode === "")
            ) {
              /* 需要密码但没密码 */
              Qmsg.error("密码不正确!");
              NetDiskUI.newAccessCodeView(
                "密码缺失",
                "jianguoyun",
                that.netDiskIndex,
                that.shareCode,
                (userInputAccessCode) => {
                  that.init(
                    that.netDiskIndex,
                    that.shareCode,
                    userInputAccessCode
                  );
                }
              );
              return;
            }
            if (fileErrorCode === "AuthenticationFailed") {
              Qmsg.error("密码错误");
              NetDiskUI.newAccessCodeView(
                void 0,
                "jianguoyun",
                that.netDiskIndex,
                that.shareCode,
                (userInputAccessCode) => {
                  that.init(
                    that.netDiskIndex,
                    that.shareCode,
                    userInputAccessCode
                  );
                }
              );
              return;
            }
            if (fileHash === "" || fileHash == void 0) {
              log.error("hash为空,可能文件被撤销分享了");
              Qmsg.error(`文件分享已被撤销`);
              return;
            }
            if (fileSize == void 0 && isdir == false) {
              log.error("无size,可能文件被删除了");
              Qmsg.error(`“${fileName}”文件已被拥有者(“${fileOwner}”)删除`);
              return;
            } else {
              return {
                name: fileName,
                hash: fileHash,
                size: fileSize,
                needsPassword: fileNeedsPassword,
                owner: fileOwner,
                isdir: isdir,
              };
            }
          } else if (
            respData.responseText.match("对不起,找不到您指定的文件。")
          ) {
            log.error("啊噢! (404) 对不起,找不到您指定的文件。");
            Qmsg.error("坚果云: 对不起,找不到您指定的文件。");
          } else if (
            respData.responseText.match("对不起,您的某些输入不正确。")
          ) {
            log.error("可能该链接不需要访问码或者访问码有问题");
            NetDiskUI.newAccessCodeView(
              void 0,
              "jianguoyun",
              that.netDiskIndex,
              that.shareCode,
              (userInputAccessCode) => {
                that.init(
                  that.netDiskIndex,
                  that.shareCode,
                  userInputAccessCode
                );
              }
            );
          } else {
            log.error("获取PageInfo失败");
            Qmsg.error("坚果云: 获取PageInfo失败");
          }
        };
        /**
         * 获取下载链接
         * @param {string} fileHash 文件hash值
         * @param {string} fileName 文件名
         * @returns {Promise}
         */
        this.getFileLink = async function (fileHash = "", fileName = "") {
          fileName = encodeURIComponent(fileName);
          let getResp = await httpx.get({
            url: `https://www.jianguoyun.com/d/ajax/fileops/pubFileLink?k=${fileHash}&name=${fileName}&wm=false${
              that.accessCode === "" ? "" : "&pd=" + that.accessCode
            }&forwin=1&_=${new Date().getTime()}`,
            responseType: "json",
            headers: {
              "User-Agent": utils.getRandomPCUA(),
            },
            onerror: function () {},
          });
          if (!getResp.status) {
            if (utils.isNotNull(getResp.data?.responseText)) {
              let errorData = utils.toJSON(getResp.data.responseText);
              log.error(["坚果云", errorData]);
              if (errorData["errorCode"] === "UnAuthorized") {
                that.gotoLogin();
              } else {
                Qmsg.error(errorData["detailMsg"]);
              }
            } else {
              Qmsg.error("请求异常");
            }
            return;
          }
          let respData = getResp.data;
          log.info(["请求信息", respData]);
          let resultJSON = utils.toJSON(respData.responseText);
          log.info(["解析JSON", resultJSON]);
          if (resultJSON.hasOwnProperty("errorCode")) {
            Qmsg.error("坚果云: " + resultJSON["detailMsg"]);
            return;
          } else if (resultJSON.hasOwnProperty("url")) {
            return resultJSON["url"];
          } else {
            Qmsg.error("坚果云: 处理下载链接异常");
          }
        };
        /**
         * 获取文件夹下的文件下载链接
         * @param {string} fileHash
         * @param {string} fileName
         * @param {string} filePath
         * @returns {Promise}
         */
        this.getDirLink = async function (
          fileHash = "",
          fileName = "",
          filePath = "/"
        ) {
          fileName = encodeURIComponent(fileName);
          let getResp = await httpx.get({
            url: `https://www.jianguoyun.com/d/ajax/dirops/pubDIRLink?k=${fileHash}&dn=${fileName}&p=${filePath}&forwin=1&_=${new Date().getTime()}`,
            responseType: "json",
            headers: {
              "User-Agent": utils.getRandomPCUA(),
            },
            onerror: function () {},
          });
          if (!getResp.status) {
            if (utils.isNotNull(getResp.data?.responseText)) {
              let errorData = utils.toJSON(getResp.data.responseText);
              log.error(["坚果云", errorData]);
              if (errorData["errorCode"] === "UnAuthorized") {
                that.gotoLogin();
              } else {
                Qmsg.error(errorData["detailMsg"]);
              }
            } else {
              Qmsg.error("请求异常");
            }
            return;
          }
          let respData = getResp.data;
          log.info(["请求信息", respData]);
          let resultJSON = utils.toJSON(respData.responseText);
          log.info(resultJSON);
          if (resultJSON.hasOwnProperty("errorCode")) {
            Qmsg.error("坚果云: " + resultJSON["detailMsg"]);
            return;
          } else if (resultJSON.hasOwnProperty("url")) {
            return resultJSON["url"];
          } else {
            Qmsg.error("坚果云: 处理下载链接异常");
          }
        };
        /**
         * 获取文件夹信息
         * @param {string} hash
         * @returns
         */
        this.getFolderInfo = async function (hash = "") {
          let getResp = await httpx.get({
            url: `https://www.jianguoyun.com/d/ajax/dirops/pubDIRBrowse?hash=${hash}&relPath=%2F&_=${new Date().getTime()}`,
            responseType: "json",
            headers: {
              "User-Agent": utils.getRandomPCUA(),
            },
          });
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          log.info(["请求信息", respData]);
          let resultJSON = utils.toJSON(respData.responseText);
          log.info(resultJSON);
          if ("objects" in resultJSON) {
            return resultJSON["objects"];
          } else {
            Qmsg.error("坚果云: 处理多文件信息异常");
          }
        };
        /**
         * 前往登录
         */
        this.gotoLogin = function () {
          NetDiskPops.confirm(
            {
              title: {
                text: "提示",
                position: "center",
              },
              content: {
                text: `解析失败,原因:当前尚未登录坚果云,是否前往登录?`,
              },
              btn: {
                reverse: true,
                position: "end",
                ok: {
                  text: "前往",
                  callback: function (_event_) {
                    window.open(
                      "https://www.jianguoyun.com/d/login#from=https%3A%2F%2Fwww.jianguoyun.com%2F",
                      "_blank"
                    );
                  },
                },
              },
            },
            NetDiskUI.popsStyle.jianGuoYunLoginTip
          );
        };
        return this;
      },
      /**
       * 奶牛快传
       * 感谢:https://github.com/qaiu/netdisk-fast-download
       * @constructor
       * @returns {object}
       */
      nainiu: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        const OK_CODE = "0000";
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          that.panelList = [];
          that.panelContent = "";
          let checkLinkValidityInfo = await that.checkLinkValidity(
            that.shareCode,
            that.accessCode
          );
          if (!checkLinkValidityInfo) {
            return;
          }
          if (checkLinkValidityInfo.isFolder) {
            /* 多文件 */
            Qmsg.info("正在递归文件");
            let QmsgLoading = Qmsg.loading(`正在解析多文件中,请稍后...`);
            let firstFolderInfo = await that.getShareFolder(
              checkLinkValidityInfo["data"]["guid"]
            );
            if (!firstFolderInfo) {
              QmsgLoading.close();
              return;
            }
            let firstFileInfo = await that.getShareFiles(
              checkLinkValidityInfo["data"]["guid"]
            );
            if (!firstFileInfo) {
              QmsgLoading.close();
              return;
            }
            let folderInfoList = that.getFolderInfo(
              checkLinkValidityInfo["data"]["guid"],
              firstFolderInfo,
              firstFileInfo,
              0
            );
            QmsgLoading.close();
            log.info("递归完毕");
            NetDiskUI.staticView.moreFile("奶牛快传文件解析", folderInfoList);
          } else {
            /* 单文件 */
            let downloadUrl = void 0;
            if (checkLinkValidityInfo["zipDownload"]) {
              downloadUrl = await that.getZipFileDownloadUrl(
                that.shareCode,
                checkLinkValidityInfo["guid"],
                checkLinkValidityInfo["fileName"]
              );
            } else {
              downloadUrl = await that.getDownloadUrl(
                that.shareCode,
                checkLinkValidityInfo["guid"],
                checkLinkValidityInfo["id"]
              );
            }
            if (!downloadUrl) {
              return;
            }
            if (NetDiskFilterScheme.isForwardDownloadLink("nainiu")) {
              downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                "nainiu",
                downloadUrl
              );
            }

            NetDiskUI.staticView.oneFile({
              title: "奶牛快传单文件直链",
              fileName: checkLinkValidityInfo["fileName"],
              fileType: checkLinkValidityInfo["fileType"],
              fileSize: checkLinkValidityInfo["fileSize"],
              downloadUrl: downloadUrl,
              fileUploadTime: checkLinkValidityInfo["fileUploadTime"],
              fileLatestTime: checkLinkValidityInfo["fileLatestTime"],
              clickCallBack: (_fileDetails_) => {
                that.downloadFile(
                  checkLinkValidityInfo["fileName"],
                  downloadUrl
                );
              },
            });
          }
        };
        /**
         * 校验链接有效性并解析获取信息
         * @param {string} shareCode
         * @param {string} accessCode
         * @param {boolean|object}
         */
        this.checkLinkValidity = async function (shareCode, accessCode) {
          let resultJSON = await that.getShareByUniqueUrl(shareCode);
          if (!resultJSON) {
            return false;
          }
          let code = resultJSON["code"];
          let message = resultJSON["message"];
          if (code !== OK_CODE) {
            Qmsg.error(message);
            return false;
          } else {
            let needPassword = resultJSON["data"]["needPassword"];
            let zipDownload = resultJSON["data"]["zipDownload"];
            if (needPassword && utils.isNull(accessCode)) {
              Qmsg.error("密码缺失!");
              NetDiskUI.newAccessCodeView(
                "密码缺失",
                "nainiu",
                that.netDiskIndex,
                that.shareCode,
                (userInputAccessCode) => {
                  that.init(
                    that.netDiskIndex,
                    that.shareCode,
                    userInputAccessCode
                  );
                }
              );
              return false;
            } else if (zipDownload) {
              /* 压缩包下载 */
              Qmsg.success("该链接为zip单文件");
              return {
                zipDownload: zipDownload,
                guid: resultJSON["data"]["guid"],
                fileSize: utils.formatByteToSize(
                  resultJSON["data"]["firstFolder"]["size"]
                ),
                fileName: resultJSON["data"]["firstFolder"]["title"],
                fileUploadTime: utils.formatTime(
                  resultJSON["data"]["firstFolder"]["created_at"]
                ),
                fileLatestTime: utils.formatTime(
                  resultJSON["data"]["firstFolder"]["updated_at"]
                ),
              };
            } else if (resultJSON["data"]["firstFile"] == void 0) {
              /* 文件夹类型 */
              Qmsg.success("该链接为文件夹类型");
              return {
                isFolder: true,
                guid: resultJSON["data"]["guid"],
                firstFolder: resultJSON["data"]["firstFolder"],
                data: resultJSON["data"],
              };
            }
            return {
              zipDownload: zipDownload,
              guid: resultJSON["data"]["guid"],
              id: resultJSON["data"]["firstFile"]["id"],
              fileSize: utils.formatByteToSize(
                resultJSON["data"]["firstFile"]["file_info"]["size"]
              ),
              fileName: resultJSON["data"]["firstFile"]["file_info"]["title"],
              fileType: resultJSON["data"]["firstFile"]["file_info"]["format"],
              fileUploadTime: utils.formatTime(
                resultJSON["data"]["firstFile"]["created_at"]
              ),
              fileLatestTime: utils.formatTime(
                resultJSON["data"]["firstFile"]["updated_at"]
              ),
            };
          }
        };
        /**
         * 获取直链弹窗的文件夹信息
         * @returns
         */
        this.getFolderInfo = function (
          transferGuid,
          shareFolderInfoList,
          shareFileInfoList,
          index = 0
        ) {
          let folderInfoList = [];
          let tempFolderInfoList = [];
          let tempFolderFileInfoList = [];
          /* 文件夹 */
          shareFolderInfoList.forEach((folderInfo) => {
            folderInfoList.push({
              fileName: folderInfo["title"],
              fileSize: 0,
              fileType: "",
              createTime: folderInfo["created_at"],
              latestTime: folderInfo["updated_at"],
              isFolder: true,
              index: index,
              async clickEvent() {
                if (
                  !folderInfo["child_folder_count"] &&
                  !folderInfo["content_count"]
                ) {
                  /* 里面没有文件夹和文件 */
                  return [];
                }
                let childFolderInfo = await that.getShareFolder(
                  transferGuid,
                  folderInfo["id"]
                );
                if (!childFolderInfo) {
                  return [];
                }
                let childFileInfo = await that.getShareFiles(
                  transferGuid,
                  folderInfo["id"]
                );
                if (!childFileInfo) {
                  return [];
                }
                let folderInfoList = that.getFolderInfo(
                  transferGuid,
                  childFolderInfo,
                  childFileInfo,
                  index + 1
                );
                return folderInfoList;
              },
            });
          });
          /* 文件 */
          shareFileInfoList.forEach((fileInfo) => {
            let fileName = fileInfo["file_info"]["title"];
            let fileType = fileInfo["file_info"]["format"] ?? "";
            if (Boolean(fileType)) {
              fileName = fileName + "." + fileType;
            }
            folderInfoList.push({
              fileName: fileName,
              fileSize: fileInfo["file_info"]["size"],
              fileType: fileType,
              createTime: fileInfo["created_at"],
              latestTime: fileInfo["updated_at"],
              isFolder: false,
              index: index,
              async clickEvent() {
                let downloadUrl = await that.getDownloadUrl(
                  that.shareCode,
                  transferGuid,
                  fileInfo["id"]
                );
                if (!downloadUrl) {
                  return;
                }
                if (NetDiskFilterScheme.isForwardDownloadLink("nainiu")) {
                  downloadUrl = NetDiskFilterScheme.parseDataToSchemeUri(
                    "nainiu",
                    downloadUrl
                  );
                }

                that.downloadFile(fileName, downloadUrl);
              },
            });
          });
          tempFolderInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          tempFolderFileInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          folderInfoList = folderInfoList.concat(tempFolderInfoList);
          folderInfoList = folderInfoList.concat(tempFolderFileInfoList);
          log.info(["getFolderInfo", folderInfoList]);
          return folderInfoList;
        };
        /**
         * 文件解析
         * @param {string} shareCode
         * @param {string} accessCode
         */
        this.parseMoreFile = async function (shareCode, accessCode) {};
        /**
         * 获取文件夹信息
         * @param {string} transferGuid
         * @param {number} folderId
         * @param {number} page
         * @param {number} size
         * @returns {Promise<?object[]>}
         */
        this.getShareFolder = async function (
          transferGuid,
          folderId = "",
          page = 0,
          size = 100
        ) {
          let getResp = await httpx.get(
            `https://cowtransfer.com/core/api/transfer/share/folders?transferGuid=${transferGuid}&folderId=${folderId}&page=${page}&size=${size}`,
            {
              headers: {
                Accept: "application/json",
                "User-Agent": utils.getRandomPCUA(),
                Referer: "https://cowtransfer.com/",
              },
            }
          );
          log.success(getResp);
          if (!getResp.status) {
            return;
          }
          let data = utils.toJSON(getResp.data.responseText);
          if (data.code !== OK_CODE) {
            Qmsg.error(data["message"]);
            return;
          }
          let folders = data["data"]["folders"];
          if (!Array.isArray(folders)) {
            Qmsg.error("data.folders不是数组");
            return;
          }
          return folders;
        };
        /**
         * 获取文件信息
         * @param {string} transferGuid
         * @param {number} folderId
         * @param {number} page
         * @param {number} size
         * @param {boolean} subContent
         * @returns {Promise<?object[]>}
         */
        this.getShareFiles = async function (
          transferGuid,
          folderId = "",
          page = 0,
          size = 20,
          subContent = false
        ) {
          let getResp = await httpx.get(
            `https://cowtransfer.com/core/api/transfer/share/files?transferGuid=${transferGuid}&folderId=${folderId}&page=${page}&size=${size}&subContent=${subContent}`,
            {
              headers: {
                Accept: "application/json",
                "User-Agent": utils.getRandomPCUA(),
                Referer: "https://cowtransfer.com/",
              },
            }
          );
          log.success(getResp);
          if (!getResp.status) {
            return;
          }
          let data = utils.toJSON(getResp.data.responseText);
          if (data.code !== OK_CODE) {
            Qmsg.error(data["message"]);
            return;
          }
          let files = data["data"]["files"];
          if (!Array.isArray(files)) {
            Qmsg.error("data.files不是数组");
            return;
          }
          return files;
        };
        /**
         * 获取分享信息
         * @param {string} shareCode
         * @returns {?{
         * code: string,
         * message: string,
         * data: {zipDownload: boolean,
         * guid:string,
         * fileSize: string,
         * fileName: string,
         * fileUploadTime: number,
         * fileLatestTime: number,
         * } | {
         * zipDownload: boolean,
         * guid:string,
         * id: string,
         * fileSize: string,
         * fileType: string,
         * fileName: string,
         * fileUploadTime: number,
         * fileLatestTime: number,
         * }[]}
         */
        this.getShareByUniqueUrl = async function (shareCode) {
          let url = `https://cowtransfer.com/core/api/transfer/share?uniqueUrl=${shareCode}`;
          let getResp = await httpx.get({
            url: url,
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Referer: "https://cowtransfer.com/s/" + shareCode,
            },
          });
          log.info(getResp);
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          let resultJSON = utils.toJSON(respData.responseText);
          log.info(["转换的JSON", resultJSON]);
          return resultJSON;
        };
        /**
         * 获取下载链接
         * @param {string} shareCode
         * @param {string} guid
         * @param {string} id
         * @returns {?string}
         */
        this.getDownloadUrl = async function (shareCode, guid = "", id = "") {
          let url = `https://cowtransfer.com/core/api/transfer/share/download?transferGuid=${guid}&fileId=${id}`;
          let getResp = await httpx.get({
            url: url,
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Referer: "https://cowtransfer.com/s/" + shareCode,
            },
          });
          log.info(getResp);
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          let resultJSON = utils.toJSON(respData.responseText);
          log.info(["转换的JSON", resultJSON]);
          if (resultJSON["code"] === OK_CODE) {
            return resultJSON["data"]["downloadUrl"];
          } else {
            Qmsg.error(`奶牛快传-获取直链:${resultJSON["message"]}`);
            return;
          }
        };
        /**
         * 获取zip文件的下载链接
         * @param {string} shareCode
         * @param {string} guid
         * @param {string} title 标题
         * @returns {?string}
         */
        this.getZipFileDownloadUrl = async function (
          shareCode,
          guid = "",
          title = ""
        ) {
          let url = `https://cowtransfer.com/core/api/transfer/share/download?transferGuid=${guid}&title=${title}`;
          let getResp = await httpx.get({
            url: url,
            headers: {
              "User-Agent": utils.getRandomPCUA(),
              Referer: "https://cowtransfer.com/s/" + shareCode,
            },
          });
          log.info(getResp);
          if (!getResp.status) {
            return;
          }
          let respData = getResp.data;
          let resultJSON = utils.toJSON(respData.responseText);
          log.info(["转换的JSON", resultJSON]);
          if (resultJSON["code"] === OK_CODE) {
            return resultJSON["data"]["downloadUrl"];
          } else {
            Qmsg.error(`奶牛快传-获取直链:${resultJSON["message"]}`);
            return;
          }
        };
        /**
         * 下载文件
         * @param {string} fileName 文件名
         * @param {string} fileDownloadUrl 下载地址
         */
        this.downloadFile = async function (fileName, fileDownloadUrl) {
          log.info(["下载文件:", fileName, fileDownloadUrl]);
          Qmsg.info(`调用【GM_download】下载:${fileName}`);
          if (typeof GM_download === "undefined") {
            Qmsg.error("当前脚本环境缺失API 【GM_download】");
            return;
          }
          let abortDownload = void 0;
          let downloadingQmsg = Qmsg.loading("下载中...", {
            showClose: true,
            onClose() {
              if (typeof abortDownload === "function") {
                abortDownload();
              }
            },
          });
          let isDownloadEnd = false;
          let GM_download_Result = GM_download({
            url: fileDownloadUrl,
            name: fileName,
            headers: {
              Referer: "https://cowtransfer.com/s/" + that.shareCode,
            },
            onload() {
              downloadingQmsg.close();
              Qmsg.success(`下载 ${fileName} 已完成`);
            },
            onprogress(details) {
              if (
                typeof details === "object" &&
                "loaded" in details &&
                "total" in details &&
                !isDownloadEnd
              ) {
                let progressNum = details.loaded / details.total;
                let formatProgressNum = (progressNum * 100).toFixed(2);
                downloadingQmsg.setText(`下载中...${formatProgressNum}%`);
                if (details.loaded === details.total) {
                  isDownloadEnd = true;
                }
              }
            },
            onerror(error) {
              downloadingQmsg.close();
              log.error(["下载失败error👉", error]);
              if (typeof error === "object" && error["error"]) {
                Qmsg.error(
                  `下载 ${fileName} 失败或已取消 原因:${error["error"]}`,
                  {
                    timeout: 6000,
                  }
                );
              } else {
                Qmsg.error(`下载 ${fileName} 失败或已取消`);
              }
            },
            ontimeout() {
              downloadingQmsg.close();
              Qmsg.error(`下载 ${fileName} 请求超时`);
            },
          });
          if (
            typeof GM_download_Result === "object" &&
            "abort" in GM_download_Result
          ) {
            abortDownload = GM_download_Result["abort"];
          }
        };
        return this;
      },
      /**
       * UC网盘
       * @constructor
       * @returns {object}
       */
      uc: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        /**
         * 入口
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode
         * @param {string} accessCode
         * @returns
         */
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          Qmsg.info("检查是否已登录UC网盘");
          let loginStatus = await that.isLogin();
          if (!Boolean(loginStatus)) {
            that.gotoLogin(
              "检测到尚未登录UC网盘,是否前去登录?<br />&nbsp;&nbsp;&nbsp;&nbsp;(注意,需要当前浏览器的UA切换成PC才有登录选项)"
            );
            return;
          }
          let stoken = await that.getStoken(that.shareCode, that.accessCode);
          if (!stoken) {
            return;
          }
          let detail = await that.getDetail(
            that.shareCode,
            that.accessCode,
            stoken
          );
          if (!detail) {
            Qmsg.error("UC网盘:获取detail失败");
            return;
          }
          if (
            detail.length === 1 &&
            detail[0].dir == false &&
            detail[0].file_type === 1
          ) {
            let oneFileDetail = detail[0];
            let oneFileDownloadDetail = await that.getDownload(
              that.shareCode,
              stoken,
              oneFileDetail.fid,
              oneFileDetail.share_fid_token
            );
            if (!oneFileDownloadDetail) {
              return;
            }
            if (!oneFileDownloadDetail[0].download_url) {
              Qmsg.error("获取download_url失败");
              return;
            }
            NetDiskUI.staticView.oneFile({
              title: "UC网盘单文件直链",
              fileName: oneFileDownloadDetail[0].file_name,
              fileSize: utils.formatByteToSize(oneFileDownloadDetail[0].size),
              downloadUrl: oneFileDownloadDetail[0].download_url,
              fileUploadTime: utils.formatTime(
                oneFileDownloadDetail[0].created_at
              ),
              fileLatestTime: utils.formatTime(
                oneFileDownloadDetail[0].last_update_at
              ),
              clickCallBack() {
                that.downloadFile(
                  oneFileDownloadDetail[0].file_name,
                  oneFileDownloadDetail[0].download_url
                );
              },
            });
          } else {
            Qmsg.info("正在递归文件");
            let QmsgLoading = Qmsg.loading(`正在解析多文件中,请稍后...`);
            let folderInfoList = that.getFolderInfo(detail, stoken, 0);
            QmsgLoading.close();
            log.info("递归完毕");
            NetDiskUI.staticView.moreFile("UC网盘文件解析", folderInfoList);
            return;
          }
        };
        /**
         * 判断是否已登录UC网盘
         * @returns {Promise<?(string|boolean)>}
         */
        this.isLogin = async function () {
          let getResp = await httpx.get("https://drive.uc.cn/", {
            headers: {
              "User-Agent": utils.getRandomPCUA(),
            },
          });
          log.success(["判断是否已登录UC网盘", getResp]);
          if (!getResp.status) {
            return;
          }
          if (getResp.data.finalUrl === "https://drive.uc.cn/list") {
            return "已登录";
          } else {
            return false;
          }
        };
        /**
         * 下载文件
         * @param {string} fileName 文件名
         * @param {string} downloadUrl 下载链接
         * @return { {
         * abort: Function
         * } }
         */
        this.downloadFile = function (fileName, downloadUrl) {
          log.info([`调用【GM_download】下载:`, arguments]);
          Qmsg.info(`调用【GM_download】下载:${fileName}`);
          if (typeof GM_download === "undefined") {
            Qmsg.error("当前脚本环境缺失API 【GM_download】");
            return;
          }
          let downloadingQmsg = Qmsg.loading("下载中...");
          let isDownloadEnd = false;
          return GM_download({
            url: downloadUrl,
            name: fileName,
            headers: {
              Referer: "https://drive.uc.cn/",
            },
            onload() {
              downloadingQmsg.close();
              Qmsg.success(`下载 ${fileName} 已完成`);
            },
            onprogress(details) {
              if (
                typeof details === "object" &&
                "loaded" in details &&
                "total" in details &&
                !isDownloadEnd
              ) {
                let progressNum = details.loaded / details.total;
                let formatProgressNum = (progressNum * 100).toFixed(2);
                downloadingQmsg.setText(`下载中...${formatProgressNum}%`);
                if (details.loaded === details.total) {
                  isDownloadEnd = true;
                }
              }
            },
            onerror(error) {
              downloadingQmsg.close();
              log.error(["下载失败error👉", error]);
              if (typeof error === "object" && error["error"]) {
                Qmsg.error(
                  `下载 ${fileName} 失败或已取消 原因:${error["error"]}`,
                  {
                    timeout: 6000,
                  }
                );
              } else {
                Qmsg.error(`下载 ${fileName} 失败或已取消`);
              }
            },
            ontimeout() {
              downloadingQmsg.close();
              Qmsg.error(`下载 ${fileName} 请求超时`);
            },
          });
        };
        /**
         * 前往登录
         * @param {string} text 弹窗的显示的内容
         */
        this.gotoLogin = function (text = "") {
          NetDiskPops.confirm(
            {
              title: {
                position: "center",
                text: "UC网盘",
              },
              content: {
                text: text,
                html: false,
              },
              btn: {
                reverse: true,
                position: "end",
                ok: {
                  text: "前往",
                  enable: true,
                  callback() {
                    window.open("https://drive.uc.cn", "_blank");
                  },
                },
              },
            },
            NetDiskUI.popsStyle.tianYiYunLoginTip
          );
        };
        /**
         * 获取stoken
         * @param {string} pwd_id 分享码
         * @param {string} passcode 访问码
         * @returns {Promise<?string>}
         */
        this.getStoken = async function (pwd_id, passcode) {
          let postResp = await httpx.post(
            "https://pc-api.uc.cn/1/clouddrive/share/sharepage/token?entry=ft&fr=pc&pr=UCBrowser",
            {
              data: JSON.stringify({
                share_for_transfer: true,
                passcode: passcode,
                pwd_id: pwd_id,
              }),
              headers: {
                Accept: "application/json, text/plain, */*",
                "Content-Type": "application/json;charset=UTF-8",
                "User-Agent": utils.getRandomPCUA(),
                Origin: "https://drive.uc.cn",
                Referer: "https://drive.uc.cn/",
              },
              onerror() {},
            }
          );
          if (!postResp.status) {
            let errorData = utils.toJSON(postResp.data.responseText);
            log.error(["获取stoken失败JSON信息", errorData]);
            if ("message" in errorData) {
              Qmsg.error(errorData["message"]);
            } else {
              Qmsg.error("请求异常,获取stoken失败");
            }
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          log.info(["获取stoken:", data]);
          if (data["code"] !== 0) {
            log.error(["获取stoken失败", data]);
            Qmsg.error("获取stoken失败");
            return;
          }
          return data["data"]["stoken"];
        };

        /**
         * 获取stoken
         * @param {string} pwd_id 分享码
         * @param {string} passcode 访问码
         * @param {string} stoken 获取的stoken
         * @param {string} [pdir_fid=0] 父fid,默认为0,如果为文件夹,那么它的fid就是这个值
         * @param {number} [force=0]
         * @param {number} [_page=1]
         * @param {number} [_size=50]
         * @param {number} [_fetch_banner=0]
         * @param {number} [_fetch_share=0]
         * @param {number} [_fetch_total=1]
         * @returns {Promise<?{
         * backup_sign: number,
         * backup_source: boolean,
         * ban: boolean,
         * category: number,
         * created_at: number,
         * creator_ucid_or_default: string,
         * cur_version_or_default: number,
         * dir: boolean,
         * duration: number,
         * event_extra: {
         *    recent_created_at: number
         * },
         * extra: string,
         * fid: string,
         * file: boolean,
         * file_name: string,
         * file_name_hl_end: number,
         * file_name_hl_start: number,
         * file_source: string,
         * file_struct: {
         *    fir_source: string,
         *    platform_source: string,
         *    sec_source: string,
         *    thi_source: string,
         *    upload_dm: string,
         *    upload_mi: string,
         * },
         * file_type: number,
         * format_type: string,
         * include_items:  number,
         * l_created_at:  number,
         * l_updated_at:  number,
         * last_update_at:  number,
         * like:  number,
         * name_space:  number,
         * offline_source: boolean,
         * operated_at:  number,
         * owner_drive_type_or_default:  number,
         * owner_ucid: string,
         * pdir_fid: string,
         * raw_name_space:  number,
         * risk_type:  number,
         * save_as_source: boolean,
         * share_fid_token: string,
         * size:  number,
         * status:  number,
         * tags: string,
         * updated_at:  number,
         * _extra: {},
         * }[]>}
         */
        this.getDetail = async function (
          pwd_id,
          passcode,
          stoken,
          pdir_fid = 0,
          force = 0,
          _page = 1,
          _size = 50,
          _fetch_banner = 0,
          _fetch_share = 0,
          _fetch_total = 1
        ) {
          let getResp = await httpx.get(
            `https://pc-api.uc.cn/1/clouddrive/transfer_share/detail?pr=UCBrowser&fr=h5&pwd_id=${pwd_id}&__t=${new Date().getTime()}&passcode=${passcode}&stoken=${encodeURIComponent(
              stoken
            )}&pdir_fid=${pdir_fid}&force=${force}&_page=${_page}&_size=${_size}&_fetch_banner=${_fetch_banner}&_fetch_share=${_fetch_share}&_fetch_total=${_fetch_total}&_sort=${encodeURIComponent(
              "file_type:asc,file_name:asc"
            )}`,
            {
              headers: {
                Accept: "application/json, text/plain, */*",
                "User-Agent": utils.getRandomPCUA(),
                Origin: "https://drive.uc.cn",
                Referer: "https://drive.uc.cn/",
              },
            }
          );
          if (!getResp.status) {
            return;
          }
          let data = utils.toJSON(getResp.data.responseText);
          log.info(["获取detail:", data]);
          if (data["code"] !== 0) {
            log.error(["获取detail失败", data]);
            Qmsg.error("获取detail失败");
            return;
          }
          return data["data"]["list"];
        };

        /**
         * 获取下载信息
         * @param {string} pwd_id 分享码
         * @param {string} stoken 获取的stoken
         * @param {string} fids 通过获取到的detail获取到的fid
         * @param {string} share_fid_token 通过获取到的detail获取到的share_fid_token
         * @returns {Promise< ?{
         * backup_sign: number,
         * backup_source: boolean,
         * ban: boolean,
         * big_thumbnail: string,
         * category: number,
         * created_at: number,
         * creator_ucid_or_default: string,
         * cur_version_or_default: number,
         * dir: boolean,
         * download_url: string,
         * duration: number,
         * event_extra: {
         *    recent_created_at: number
         * },
         * extra: string,
         * fid: string,
         * file: boolean,
         * file_name: string,
         * file_name_hl_end: number,
         * file_name_hl_start: number,
         * file_source: string,
         * file_type: number,
         * format_type: string,
         * l_created_at: number,
         * l_updated_at: number,
         * last_update_at: number,
         * like: number,
         * md5: string,
         * name_space: number,
         * obj_category: string,
         * offline_source: boolean,
         * operated_at: number,
         * owner_drive_type_or_default: number,
         * owner_ucid: string,
         * pdir_fid: string,
         * preview_url: string,
         * range_size: number,
         * raw_name_space: number,
         * risk_type: number,
         * save_as_source: boolean,
         * share_fid_token: string,
         * size: number,
         * status: number,
         * thumbnail: string,
         * updated_at: number,
         * video_height: number,
         * video_max_resolution: string,
         * video_width: number,
         * _extra: {},
         * } []>}
         */
        this.getDownload = async function (
          pwd_id,
          stoken,
          fid,
          share_fid_token
        ) {
          let postResp = await httpx.post(
            "https://pc-api.uc.cn/1/clouddrive/file/download?entry=ft&fr=pc&pr=UCBrowser",
            {
              data: JSON.stringify({
                fids: [fid],
                pwd_id: pwd_id,
                stoken: stoken,
                fids_token: [share_fid_token],
              }),
              headers: {
                Accept: "application/json, text/plain, */*",
                "Content-Type": "application/json;charset=UTF-8",
                "User-Agent": utils.getRandomPCUA(),
                Origin: "https://drive.uc.cn",
                Referer: "https://drive.uc.cn/",
              },
            }
          );
          if (!postResp.status) {
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          log.info(["获取download:", data]);
          if (data["code"] !== 0) {
            log.error(["获取download失败", data]);
            Qmsg.error("获取download失败");
            return;
          }
          if (data["data"].length === 0) {
            log.error(["获取download detail失败", data]);
            Qmsg.error("获取download detail失败失败");
            return;
          }
          return data["data"];
        };

        /**
         * 获取文件夹信息
         * @param {{
         * backup_sign: number,
         * backup_source: boolean,
         * ban: boolean,
         * category: number,
         * created_at: number,
         * creator_ucid_or_default: string,
         * cur_version_or_default: number,
         * dir: boolean,
         * duration: number,
         * event_extra: {
         *    recent_created_at: number
         * },
         * extra: string,
         * fid: string,
         * file: boolean,
         * file_name: string,
         * file_name_hl_end: number,
         * file_name_hl_start: number,
         * file_source: string,
         * file_struct: {
         *    fir_source: string,
         *    platform_source: string,
         *    sec_source: string,
         *    thi_source: string,
         *    upload_dm: string,
         *    upload_mi: string,
         * },
         * file_type: number,
         * format_type: string,
         * include_items:  number,
         * l_created_at:  number,
         * l_updated_at:  number,
         * last_update_at:  number,
         * like:  number,
         * name_space:  number,
         * offline_source: boolean,
         * operated_at:  number,
         * owner_drive_type_or_default:  number,
         * owner_ucid: string,
         * pdir_fid: string,
         * raw_name_space:  number,
         * risk_type:  number,
         * save_as_source: boolean,
         * share_fid_token: string,
         * size:  number,
         * status:  number,
         * tags: string,
         * updated_at:  number,
         * _extra: {},
         * }[]} infoList
         * @return {Promise<{
         * fileName: string,
         * fileSize: string|number,
         * fileType: ?string,
         * createTime: ?string,
         * latestTime: ?string,
         * isFolder: boolean,
         * index: ?number,
         * clickCallBack: ?(event:Event,_config_: object)=>{}
         * }[]>}
         */
        this.getFolderInfo = function (infoList, stoken, index = 0) {
          let folderInfoList = [];
          let tempFolderInfoList = [];
          /**
           * @type {PopsFolderDataConfig[]}
           */
          let tempFolderFileInfoList = [];
          infoList.forEach((item) => {
            if (item.dir == false && item.file_type === 1) {
              /* 文件 */
              tempFolderFileInfoList.push({
                fileName: item.file_name,
                fileSize: item.size,
                fileType: "",
                createTime: item.created_at,
                latestTime: item.updated_at,
                isFolder: false,
                index: index,
                async clickEvent() {
                  let fileDownloadUrl = await that.getDownload(
                    that.shareCode,
                    stoken,
                    item.fid,
                    item.share_fid_token
                  );
                  if (fileDownloadUrl) {
                    if (fileDownloadUrl.length) {
                      fileDownloadUrl = fileDownloadUrl[0].download_url;
                    } else {
                      fileDownloadUrl = "";
                    }
                  } else {
                    fileDownloadUrl = "";
                  }
                  if (item.ban) {
                    Qmsg.error("文件已被禁止下载");
                  } else {
                    let schemeDownloadUrl = fileDownloadUrl;
                    if (NetDiskFilterScheme.isForwardDownloadLink("uc")) {
                      schemeDownloadUrl =
                        NetDiskFilterScheme.parseDataToSchemeUri(
                          "uc",
                          schemeDownloadUrl
                        );
                    }
                    /* 如果已被scheme过滤,那么不进行GM_download下载 */
                    if (schemeDownloadUrl === fileDownloadUrl) {
                      that.downloadFile(item.file_name, fileDownloadUrl);
                    } else {
                      return {
                        autoDownload: true,
                        mode: "aBlank",
                        url: fileDownloadUrl,
                      };
                    }
                  }
                },
              });
            } else {
              /* 文件夹 */
              tempFolderInfoList.push({
                fileName: item.file_name,
                fileSize: item.size,
                fileType: "",
                createTime: item.created_at,
                latestTime: item.updated_at,
                isFolder: true,
                index: index,
                async clickEvent() {
                  if (item.include_items === 0) {
                    /* 里面没有文件 */
                    log.success("里面没有文件");
                    return [];
                  }
                  let newDetail = await that.getDetail(
                    that.shareCode,
                    that.accessCode,
                    stoken,
                    item.fid
                  );
                  if (newDetail) {
                    return that.getFolderInfo(newDetail, stoken, index + 1);
                  } else {
                    return [];
                  }
                },
              });
            }
          });

          tempFolderInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          tempFolderFileInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          folderInfoList = folderInfoList.concat(tempFolderInfoList);
          folderInfoList = folderInfoList.concat(tempFolderFileInfoList);
          log.info(["getFilesInfoByRec", folderInfoList]);
          return folderInfoList;
        };
        return this;
      },
      /**
       * 阿里云盘
       * @constructor
       * @returns {object}
       */
      aliyun: function () {
        let that = this;
        /**
         * 所在规则的下标
         */
        this.netDiskIndex = 0;
        /**
         * 分享码
         */
        this.shareCode = "";
        /**
         * 提取码
         */
        this.accessCode = "";
        this.X_Share_Token_Data = {
          expire_time: "2000-01-01T00:00:00.000Z",
          expires_in: 7200,
          share_token: "",
        };
        /**
         * header请求头 X-Device-Id
         */
        this.X_Device_Id = null;
        /**
         * header请求头 X-Canary
         */
        this.X_Canary = "client=web,app=share,version=v2.3.1";
        /**
         * 入口
         * @param {number} netDiskIndex 网盘名称索引下标
         * @param {string} shareCode
         * @param {string} accessCode
         * @returns
         */
        this.init = async function (netDiskIndex, shareCode, accessCode) {
          log.info([netDiskIndex, shareCode, accessCode]);
          that.netDiskIndex = netDiskIndex;
          that.shareCode = shareCode;
          that.accessCode = accessCode;
          that.X_Device_Id = that.get_X_Device_Id();
          log.info("生成X_Device_Id:" + that.X_Device_Id);
          if (
            globalThis.location.hostname !== "www.aliyundrive.com" &&
            globalThis.location.hostname !== "www.alipan.com"
          ) {
            let url = NetDiskParse.getBlankUrl(
              "aliyun",
              that.netDiskIndex,
              that.shareCode,
              that.accessCode
            );
            let errorTip = Qmsg.error(
              `请在阿里云盘页面解析,<a href="${url}">点我前往</a>`,
              {
                html: true,
                timeout: 10000,
              }
            );
            DOMUtils.on(
              errorTip.$elem.querySelector("a[href]"),
              "click",
              void 0,
              (event) => {
                utils.preventEvent(event);
                NetDiskParse.blank(
                  url,
                  "aliyun",
                  that.netDiskIndex,
                  that.shareCode,
                  that.accessCode
                );
              }
            );
            return;
          }
          let detail = await this.list_by_share(shareCode, "root");
          if (!detail) {
            return;
          }
          Qmsg.info("正在解析链接");
          let QmsgLoading = Qmsg.loading(`正在解析多文件中,请稍后...`);
          let folderInfoList = that.getFolderInfo(detail, 0);
          QmsgLoading.close();
          log.info("解析完毕");
          NetDiskUI.staticView.moreFile("阿里云盘文件解析", folderInfoList);
        };
        /**
         * 弹窗使用-获取文件夹信息
         * @param {{
         * category?: string,
         * domain_id?: string,
         * file_extension?: string,
         * mime_extension?: string,
         * mime_type?: string,
         * punish_flag: number,
         * created_at: string,
         * domain_id: string,
         * drive_id: string,
         * file_id: string,
         * name: string,
         * parent_file_id:string,
         * share_id: string,
         * type: string,
         * updated_at: string,
         * }[]} infoList
         * @return {Promise<{
         * fileName: string,
         * fileSize: string|number,
         * fileType: ?string,
         * createTime: ?string,
         * latestTime: ?string,
         * isFolder: boolean,
         * index: ?number,
         * clickCallBack: ?(event:Event,_config_: object)=>{}
         * }[]>}
         */
        this.getFolderInfo = function (infoList, index = 0) {
          let folderInfoList = [];
          let tempFolderInfoList = [];
          /**
           * @type {PopsFolderDataConfig[]}
           */
          let tempFolderFileInfoList = [];
          infoList.forEach((item) => {
            if (item.type !== "folder") {
              /* 文件 */
              tempFolderFileInfoList.push({
                fileName: item.name,
                fileSize: item.size,
                fileType: item.file_extension,
                createTime: new Date(item.created_at).getTime(),
                latestTime: new Date(item.updated_at).getTime(),
                isFolder: false,
                index: index,
                async clickEvent() {
                  let fileDownloadUrl = await that.get_share_link_download_url(
                    item.share_id,
                    item.file_id
                  );
                  if (!fileDownloadUrl) {
                    return;
                  }
                  let schemeDownloadUrl = fileDownloadUrl;
                  if (NetDiskFilterScheme.isForwardDownloadLink("aliyun")) {
                    schemeDownloadUrl =
                      NetDiskFilterScheme.parseDataToSchemeUri(
                        "aliyun",
                        schemeDownloadUrl
                      );
                  }
                  return {
                    autoDownload: true,
                    mode: "aBlank",
                    url: schemeDownloadUrl,
                  };
                },
              });
            } else {
              /* 文件夹 */
              tempFolderInfoList.push({
                fileName: item.name,
                fileSize: 0,
                fileType: "",
                createTime: item.created_at,
                latestTime: item.updated_at,
                isFolder: true,
                index: index,
                async clickEvent() {
                  let newDetail = await that.list_by_share(
                    item.share_id,
                    item.file_id
                  );
                  if (newDetail) {
                    return that.getFolderInfo(newDetail, index + 1);
                  } else {
                    return [];
                  }
                },
              });
            }
          });

          tempFolderInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          tempFolderFileInfoList.sort((a, b) =>
            a["fileName"].localeCompare(b["fileName"])
          );
          folderInfoList = folderInfoList.concat(tempFolderInfoList);
          folderInfoList = folderInfoList.concat(tempFolderFileInfoList);
          log.info(["getFilesInfoByRec", folderInfoList]);
          return folderInfoList;
        };
        /**
         * 列出文件列表
         * @param {string} share_id
         * @param {string} parent_file_id 父项,根是root
         * @param {"name"} order_by 根据xxx排序
         * @param {"ASC"|"DESC"} order_direction 排序规则(升序/降序)
         * @returns {Promise<{
         * category?: string,
         * domain_id?: string,
         * file_extension?: string,
         * mime_extension?: string,
         * mime_type?: string,
         * punish_flag: number,
         * created_at: string,
         * domain_id: string,
         * drive_id: string,
         * file_id: string,
         * name: string,
         * parent_file_id:string,
         * share_id: string,
         * type: string,
         * updated_at: string,
         * }[]>}
         */
        this.list_by_share = async function (
          share_id,
          parent_file_id,
          order_by = "name",
          order_direction = "DESC"
        ) {
          let postResp = await httpx.post(
            "https://api.aliyundrive.com/adrive/v2/file/list_by_share",
            {
              data: JSON.stringify({
                share_id: share_id,
                parent_file_id: parent_file_id,
                limit: 20,
                image_thumbnail_process: "image/resize,w_256/format,jpeg",
                image_url_process:
                  "image/resize,w_1920/format,jpeg/interlace,1",
                video_thumbnail_process:
                  "video/snapshot,t_1000,f_jpg,ar_auto,w_256",
                order_by: order_by,
                order_direction: order_direction,
              }),
              headers: {
                Accept: "application/json, text/plain, */*",
                "Content-Type": "application/json",
                Origin: "https://www.aliyundrive.com",
                Referer: "https://www.aliyundrive.com/",
                "X-Canary": that.X_Canary,
                "X-Device-Id": that.X_Device_Id,
                "X-Share-Token": await that.get_X_Share_Token(
                  that.shareCode,
                  that.accessCode
                ),
                "User-Agent": utils.getRandomPCUA(),
              },
              onerror() {},
            }
          );
          if (!postResp.status) {
            that.handle_request_error(postResp);
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          log.info(["列出文件列表:", data]);
          return data["items"];
        };
        /**
         * 获取文件的下载链接
         * @returns {Promise<string>}
         */
        this.get_share_link_download_url = async function (share_id, file_id) {
          let postResp = await httpx.post(
            "https://api.aliyundrive.com/v2/file/get_share_link_download_url",
            {
              data: JSON.stringify({
                expire_sec: 600,
                file_id: file_id,
                share_id: share_id,
              }),
              headers: {
                Accept: "application/json, text/plain, */*",
                "Content-Type": "application/json",
                Origin: "https://www.aliyundrive.com",
                Referer: "https://www.aliyundrive.com/",
                "Content-Type": "application/json;charset=UTF-8",
                Authorization: "Bearer " + that.getAuthorization(),
                "X-Share-Token": await that.get_X_Share_Token(
                  that.shareCode,
                  that.accessCode
                ),
                "User-Agent": utils.getRandomPCUA(),
              },
              onerror() {},
            }
          );
          if (!postResp.status) {
            that.handle_request_error(postResp);
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          log.info(["获取文件的下载链接:", data]);
          return data["download_url"];
        };
        /**
         * 处理请求的错误
         * @param {HttpxAsyncResult} postResp
         */
        this.handle_request_error = function (postResp) {
          log.error(postResp);
          let errData = utils.toJSON(postResp.data.responseText);
          Qmsg.error(errData["message"]);
        };
        /**
         * 获取用户鉴权值
         * 来源:localStorage => token.access_token
         */
        this.getAuthorization = function () {
          let token = unsafeWindow.localStorage.getItem("token");
          if (utils.isNotNull(token)) {
            token = utils.toJSON(token);
            let access_token = token["access_token"];
            log.success(["获取阿里云盘的access_token:", access_token]);
            return access_token;
          } else {
            log.error("获取access_token失败,请先登录账号!");
            Qmsg.error("获取access_token失败,请先登录账号!");
          }
        };
        /**
         * 获取header请求头 X-Device-Id
         */
        this.get_X_Device_Id = function () {
          for (
            var alipan_device_id_pattern =
                /^(?:[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}|00000000-0000-0000-0000-000000000000)$/i,
              alipan_s = [],
              alipan_l = 0;
            alipan_l < 256;
            ++alipan_l
          )
            alipan_s.push((alipan_l + 256).toString(16).substr(1));
          function alipan_o() {
            return crypto.getRandomValues(new Uint8Array(16));
          }
          var alipan_c = function (args_e) {
              var second_arg =
                  arguments.length > 1 && void 0 !== arguments[1]
                    ? arguments[1]
                    : 0,
                devices_id_string = (
                  alipan_s[args_e[second_arg + 0]] +
                  alipan_s[args_e[second_arg + 1]] +
                  alipan_s[args_e[second_arg + 2]] +
                  alipan_s[args_e[second_arg + 3]] +
                  "-" +
                  alipan_s[args_e[second_arg + 4]] +
                  alipan_s[args_e[second_arg + 5]] +
                  "-" +
                  alipan_s[args_e[second_arg + 6]] +
                  alipan_s[args_e[second_arg + 7]] +
                  "-" +
                  alipan_s[args_e[second_arg + 8]] +
                  alipan_s[args_e[second_arg + 9]] +
                  "-" +
                  alipan_s[args_e[second_arg + 10]] +
                  alipan_s[args_e[second_arg + 11]] +
                  alipan_s[args_e[second_arg + 12]] +
                  alipan_s[args_e[second_arg + 13]] +
                  alipan_s[args_e[second_arg + 14]] +
                  alipan_s[args_e[second_arg + 15]]
                ).toLowerCase();
              if (
                !(function (e) {
                  return (
                    "string" == typeof e && alipan_device_id_pattern.test(e)
                  );
                })(devices_id_string)
              )
                throw TypeError("Stringified UUID is invalid");
              return devices_id_string;
            },
            alipan_u = function (args_e, args_t, args_n) {
              var randomValue =
                (args_e = args_e || {}).random || (args_e.rng || alipan_o)();
              if (
                ((randomValue[6] = (15 & randomValue[6]) | 64),
                (randomValue[8] = (63 & randomValue[8]) | 128),
                args_t)
              ) {
                args_n = args_n || 0;
                for (var i = 0; i < 16; ++i)
                  args_t[args_n + i] = randomValue[i];
                return args_t;
              }
              return alipan_c(randomValue);
            };
          return alipan_u();
        };
        /**
         * 获取header请求头 X-Share-Token
         * 来源:localStorage => shareToken.share_token
         */
        this.get_X_Share_Token = async function (share_id, share_pwd) {
          if (new Date() < new Date(that.X_Share_Token_Data.expire_time)) {
            return that.X_Share_Token_Data.share_token;
          }
          let postResp = await httpx.post(
            "https://api.aliyundrive.com/v2/share_link/get_share_token",
            {
              data: JSON.stringify({
                share_id: share_id,
                share_pwd: share_pwd,
              }),
              headers: {
                Accept: "application/json, text/plain, */*",
                "Content-Type": "application/json",
                Origin: "https://www.aliyundrive.com",
                Referer: "https://www.aliyundrive.com/",
                "X-Canary": that.X_Canary,
                "X-Device-Id": that.X_Device_Id,
                "User-Agent": utils.getRandomPCUA(),
              },
              onerror() {},
            }
          );
          if (!postResp.status) {
            that.handle_request_error(postResp);
            return;
          }
          let data = utils.toJSON(postResp.data.responseText);
          that.X_Share_Token_Data = data;
          log.info(["获取share_token:", that.X_Share_Token_Data]);
          return that.X_Share_Token_Data["share_token"];
        };
        return this;
      },
    },
    /**
     * 网盘链接解析
     * @param {string} netDiskName 网盘名称
     * @param {number} netDiskIndex 网盘名称索引下标
     * @param {string} shareCode
     * @param {string} accessCode
     */
    async parse(netDiskName, netDiskIndex, shareCode, accessCode) {
      Qmsg.info("正在获取直链");
      if (this.netDisk[netDiskName]) {
        let parseObj = new NetDiskParse.netDisk[netDiskName]();
        await parseObj.init(netDiskIndex, shareCode, accessCode);
      } else {
        log.error(`${netDiskName} 不存在解析`);
        Qmsg.error("该链接不存在解析功能");
      }
    },
    /**
     * 复制到剪贴板
     * @param {string} netDiskName
     * @param {number} netDiskIndex
     * @param {string} shareCode
     * @param {string} accessCode
     * @param {string} toastText 提示的文字
     */
    copyText(
      netDiskName,
      netDiskIndex,
      shareCode,
      accessCode,
      toastText = "已复制"
    ) {
      utils.setClip(
        NetDiskParse.getCopyUrlInfo(
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode
        )
      );
      Qmsg.success(toastText);
    },
    /**
     * 新标签页打开
     * @param {string} url 跳转的网址
     * @param {string} netDiskName
     * @param {number|string} netDiskIndex
     * @param {string} shareCode
     * @param {string} accessCode
     */
    blank(url, netDiskName, netDiskIndex, shareCode, accessCode) {
      log.success(["新标签页打开", [...arguments]]);
      if (NetDiskAutoFillAccessCode.$data.enable) {
        NetDiskAutoFillAccessCode.setValue({
          url: url,
          netDiskName: netDiskName,
          netDiskIndex: netDiskIndex,
          shareCode: shareCode,
          accessCode: accessCode,
        });
      }
      if (NetDiskFilterScheme.isForwardBlankLink(netDiskName)) {
        url = NetDiskFilterScheme.parseDataToSchemeUri(netDiskName, url);
      }
      /* 百度网盘会拒绝referrer不安全访问 */
      document
        .querySelector("meta[name='referrer']")
        ?.setAttribute("content", "no-referrer");
      if (
        utils.isNotNull(accessCode) &&
        GM_getValue(`${netDiskName}-open-blank-with-copy-accesscode`)
      ) {
        /* 等待复制完毕再跳转 */
        utils.setClip(accessCode).then(() => {
          window.open(url, "_blank");
        });
      } else {
        window.open(url, "_blank");
      }
    },
    /**
     * 将链接转为Scheme格式并打开
     * @param {string} netDiskName 网盘名称
     * @param {number} netDiskIndex 网盘名称索引下标
     * @param {string} shareCode
     * @param {string} accessCode
     */
    openScheme(netDiskName, netDiskIndex, shareCode, accessCode) {
      let url = NetDiskParse.getBlankUrl(
        netDiskName,
        netDiskIndex,
        shareCode,
        accessCode
      );
      url = NetDiskFilterScheme.parseDataToSchemeUri(netDiskName, url);
      window.open(url, "_blank");
    },
    /**
     * 获取重定向后的直链
     * @param {string} url
     * @param {string} userAgent 用户代理字符串
     * @returns {Promise}
     */
    async getRedirectFinalUrl(url, userAgent) {
      if (!GM_getValue("getTheDirectLinkAfterRedirection", true)) {
        return url;
      }
      Qmsg.success("获取重定向后的直链");
      log.info("开始获取重定向后的直链");
      let headResp = await httpx.head({
        url: url,
        headers: {
          "User-Agent": userAgent,
          Referer: window.location.origin,
        },
      });
      if (headResp.status) {
        return headResp.data.finalUrl;
      } else {
        return url;
      }
    },
    /**
     * 获取用于跳转的url
     * @param {string} netDiskName
     * @param {number} netDiskIndex
     * @param {?string} shareCode
     * @param {?string} accessCode
     * @returns {string}
     */
    getBlankUrl(netDiskName, netDiskIndex, shareCode, accessCode) {
      /**
       * @type {NetDiskRegularOption}
       */
      let regularOption = NetDisk.regular[netDiskName][netDiskIndex];
      let blankUrl = regularOption.blank;
      if (shareCode) {
        blankUrl = NetDisk.replaceParam(blankUrl, {
          shareCode: shareCode,
        });
      }
      if (accessCode && accessCode !== "") {
        blankUrl = NetDisk.replaceParam(blankUrl, {
          accessCode: accessCode,
        });
      } else {
        blankUrl = blankUrl.replace(NetDisk.noAccessCodeRegExp, "");
      }
      /**
       * 当前字典
       */
      let currentDict = NetDisk.linkDict.get(netDiskName).get(shareCode);
      if (regularOption.paramMatch) {
        let paramMatchArray = currentDict.matchText.match(
          regularOption.paramMatch
        );
        let replaceParamData = {};
        for (let index = 0; index < paramMatchArray.length; index++) {
          replaceParamData[`$${index}`] = paramMatchArray[index];
        }
        blankUrl = NetDisk.replaceParam(blankUrl, replaceParamData);
      }
      return blankUrl;
    },
    /**
     * 获取用于复制到剪贴板的网盘信息
     * @param {string} netDiskName
     * @param {number} netDiskIndex
     * @param {?string} shareCode
     * @param {?string} accessCode
     * @returns {string}
     */
    getCopyUrlInfo(netDiskName, netDiskIndex, shareCode, accessCode) {
      /**
       * @type {NetDiskRegularOption}
       */
      let regularOption = NetDisk.regular[netDiskName][netDiskIndex];
      let copyUrl = regularOption["copyUrl"];
      if (shareCode) {
        copyUrl = NetDisk.replaceParam(copyUrl, {
          shareCode: shareCode,
        });
      }
      if (accessCode && accessCode !== "") {
        copyUrl = NetDisk.replaceParam(copyUrl, {
          accessCode: accessCode,
        });
      } else {
        copyUrl = copyUrl.replace(NetDisk.noAccessCodeRegExp, "");
      }
      /**
       * 当前字典
       */
      let currentDict = NetDisk.linkDict.get(netDiskName).get(shareCode);
      if (regularOption.paramMatch) {
        let paramMatchArray = currentDict.matchText.match(
          regularOption.paramMatch
        );
        let replaceParamData = {};
        for (let index = 0; index < paramMatchArray.length; index++) {
          replaceParamData[`$${index}`] = paramMatchArray[index];
        }
        copyUrl = NetDisk.replaceParam(copyUrl, replaceParamData);
      }
      return copyUrl;
    },
  };

  /** 网盘-直链鉴权获取处理 */
  const NetDiskAuthorization = {
    /**
     * 运行于ready
     */
    init() {
      Object.keys(NetDiskAuthorization.netDisk).forEach((keyName) => {
        this.netDisk[keyName]();
      });
    },
    netDisk: {
      /**
       * 123网盘,一般用于>100MB的文件直链获取
       */
      _123pan() {
        if (window.location.hostname !== "www.123pan.com") {
          return;
        }
        /* 没在设置中开启直链获取就不获取鉴权信息 */
        if (!GM_getValue("_123pan-static-enable")) {
          return;
        }
        let authorToken = unsafeWindow.localStorage.getItem("authorToken");
        if (utils.isNull(authorToken)) {
          return;
        }
        /* 去除左右的引号 */
        authorToken = authorToken.replace(/^\"/, "").replace(/\"$/, "");
        log.success("获取123网盘已登录用户的authorToken值👇");
        log.success(authorToken);
        GM_setValue("_123pan_User_Authorization", authorToken);
      },
      /**
       * 蓝奏优选
       */
      lanzouyx() {
        /* 目前uuid可生成,userId可以通过请求获取到,暂不需要获取本地存储的 */
        return;
        if (window.location.hostname !== "www.ilanzou.com") {
          return;
        }
        /* 没在设置中开启直链获取就不获取鉴权信息 */
        if (!GM_getValue("lanzouyx-static-enable")) {
          return;
        }
        const uuid = unsafeWindow.localStorage.getItem("uuid");
        let diskVueX = unsafeWindow.localStorage.getItem("disk-vuex");
        let diskPCVueX = unsafeWindow.localStorage.getItem("disk-pc-vuex");
        if (utils.isNotNull(uuid)) {
          log.success(["获取蓝奏云优享生成的uuid:", uuid]);
          GM_setValue("lanzouyx-uuid", uuid);
        }
        if (utils.isNotNull(diskVueX)) {
          diskVueX = utils.toJSON(diskVueX);
          let userId = diskVueX?.["admin"]?.["account"]?.["info"]?.["userId"];
          if (userId != null) {
            log.success(["获取蓝奏云优享已登录用户的userId:", userId]);
            GM_setValue("lanzouyx-userId", userId);
          }
        }
        if (utils.isNotNull(diskPCVueX)) {
          diskPCVueX = utils.toJSON(diskPCVueX);
          let userId = diskVueX?.["common"]?.["accountInfo"]?.["userId"];
          if (userId != null) {
            log.success(["获取蓝奏云优享已登录用户的userId:", userId]);
            GM_setValue("lanzouyx-userId", userId);
          }
        }
      },
    },
  };

  /** 网盘-直链进行Scheme过滤 */
  const NetDiskFilterScheme = {
    protocol: "jumpwsv",
    pathname: "go",
    /**
     * 处理链接
     * @param {string} name 规则名
     * @param {string} intentData 需要处理的数据
     * @returns {string}
     */
    parseDataToSchemeUri(name, intentData) {
      /** 是否启用 */
      let isEnable = GM_getValue(name + "-forward-scheme-enable", false);
      if (!isEnable) {
        return intentData;
      }
      /** 转发的scheme */
      let schemeUri = GM_getValue(name + "-static-scheme-uri");
      if (utils.isNull(schemeUri)) {
        schemeUri = this.getSchemeUri(this.getIDMSchemeUriOption(intentData));
      }
      if (schemeUri.startsWith(this.protocol)) {
        intentData = intentData.replace(/&/g, "{-and-}");
        intentData = intentData.replace(/#/g, "{-number-}");
      }
      schemeUri = NetDisk.replaceParam(schemeUri, {
        intentData: intentData,
      });
      return schemeUri;
    },
    isForwardDownloadLink(key) {
      return GM_getValue(`${key}-forward-download-link-enable`, false);
    },
    isForwardBlankLink(key) {
      return GM_getValue(`${key}-forward-blank-link-enable`, false);
    },
    getSchemeUri(option) {
      return `${this.protocol}://${this.pathname}?package=${option["package"]}&activity=${option["activity"]}&intentAction=${option["intentAction"]}&intentData=${option["intentData"]}&intentExtra=${option["intentExtra"]}`;
    },
    getIDMSchemeUriOption(intentData = "") {
      return {
        package: "idm.internet.download.manager.plus",
        activity: "idm.internet.download.manager.UrlHandlerDownloader",
        intentAction: "android.intent.action.VIEW",
        intentData: intentData,
        intentExtra: "",
      };
    },
  };

  /** 网盘-自动填入访问码 */
  const NetDiskAutoFillAccessCode = {
    key: "tempNetDiskInfo",
    $data: {
      /**
       * 当前的网盘数据
       * @type {?NetDiskAutoFillAccessCodeOption}
       */
      netDiskInfo: null,
      /**
       * @type {boolean} 自动输入访问码是否开启
       */
      enable: Boolean(GM_getValue("autoFillAccessCode")),
    },
    /**
     * 初始化
     */
    init() {
      this.$data.netDiskInfo = this.getValue();
      if (!this.$data.netDiskInfo) {
        return;
      }
      if (!this.$data.enable) {
        return;
      }
      /* 访问码为空的话就不填 */
      if (utils.isNull(this.$data.netDiskInfo.accessCode)) {
        return;
      }
      /* 百度如果shareCode第一位是1的话,新版本会在href中去除这个1 */
      if (
        this.$data.netDiskInfo.netDiskName === "baidu" &&
        this.$data.netDiskInfo.shareCode.startsWith("1")
      ) {
        if (
          !window.location.href.includes(
            this.$data.netDiskInfo.shareCode.slice(
              1,
              this.$data.netDiskInfo.shareCode.length
            )
          )
        ) {
          return;
        }
      } else if (
        !window.location.href.includes(this.$data.netDiskInfo.shareCode)
      ) {
        return;
      }
      if (
        this.$data.netDiskInfo.netDiskName in NetDiskAutoFillAccessCode.netDisk
      ) {
        NetDiskAutoFillAccessCode.netDisk[this.$data.netDiskInfo.netDiskName](
          this.$data.netDiskInfo
        );
      }
    },
    netDisk: {
      /**
       * 百度网盘
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      baidu(netDiskInfo) {
        if (
          window.location.hostname === "pan.baidu.com" &&
          window.location.pathname === "/share/init" &&
          window.location.search.startsWith("?surl=")
        ) {
          log.success(["自动填写链接", netDiskInfo]);
          utils.waitNode("div.verify-form #accessCode").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.dispatchEvent(element, "input");
            document.querySelector("div.verify-form #submitBtn")?.click();
          });
        }
        if (
          window.location.hostname === "pan.baidu.com" &&
          window.location.pathname === "/wap/init" &&
          window.location.search.startsWith("?surl=")
        ) {
          log.success(["自动填写链接", netDiskInfo]);
          utils
            .waitNode(
              "div.extractWrap div.extract-content div.extractInputWrap.extract input[type=text]"
            )
            .then((inputElement) => {
              if (!utils.isVisible(inputElement)) {
                log.error("输入框不可见,不输入密码");
                return;
              }
              Qmsg.success("自动填入访问码");
              inputElement.value = netDiskInfo.accessCode;
              utils.dispatchEvent(inputElement, "input");
              document
                .querySelector(
                  "div.extractWrap div.extract-content button.m-button"
                )
                ?.click();
            });
        }
      },
      /**
       * 蓝奏云
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      lanzou(netDiskInfo) {
        if (window.location.hostname.match(/lanzou[a-z]{1}.com/gi)) {
          log.success(["自动填写链接", netDiskInfo]);
          utils.waitNode("#pwd").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.dispatchEvent(element, "input");
            (
              document.querySelector("#passwddiv div.passwddiv-input > div") ||
              element.nextElementSibling
            )?.click();
            document.querySelector("#sub")?.click();
          });
          utils.waitNode("#f_pwd").then((element) => {
            utils.mutationObserver(element, {
              config: {
                attributes: true,
                attributeFilter: ["style"],
              },
              callback: (mutations, observer) => {
                let inputElement = document.querySelector("#f_pwd #pwd");
                if (!utils.isVisible(inputElement)) {
                  log.error("输入框不可见,不输入密码");
                  return;
                }
                observer.disconnect();
                log.success("自动填入访问码并关闭观察者");
                Qmsg.success("自动填入访问码");
                inputElement.value = netDiskInfo.accessCode;
                utils.dispatchEvent(inputElement, "input");
                document.querySelector("#f_pwd #sub")?.click();
              },
            });
          });
        }
      },
      /**
       * 天翼云
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      tianyiyun(netDiskInfo) {
        if (window.location.hostname === "cloud.189.cn") {
          /* 桌面端 */
          log.success(["自动填写链接", netDiskInfo]);
          /**
           * 循环等待元素出现
           * @param {HTMLElement} targetElement
           * @param {Function} callback
           */
          function loopWaitElementShow(targetElement, callback) {
            let loopCount = 0;
            let maxLoopCount = 30;
            let interval = setInterval(() => {
              loopCount++;
              if (loopCount > maxLoopCount) {
                log.error("结束循环检查,退出。");
                clearInterval(interval);
                return;
              }
              if (!utils.isVisible(targetElement)) {
                log.warn(`第 ${loopCount} 次:输入框不可见,不输入密码`);
                return;
              }
              callback();
              clearInterval(interval);
            }, 500);
          }

          utils.waitNode("input#code_txt").then((codeTxtElement) => {
            loopWaitElementShow(codeTxtElement, () => {
              Qmsg.success("自动填入访问码");
              let visitBtn = document.querySelector(".btn.btn-primary.visit");
              codeTxtElement.value = netDiskInfo.accessCode;
              codeTxtElement._value = netDiskInfo.accessCode;
              utils.dispatchEvent(codeTxtElement, "input");
              utils.dispatchEvent(visitBtn, "click");
            });
          });
        }
        if (window.location.hostname === "h5.cloud.189.cn") {
          /* 移动端 */
          log.success(["自动填写链接", netDiskInfo]);
          utils
            .waitNode("input.access-code-input")
            .then((accessInputElement) => {
              loopWaitElementShow(accessInputElement, () => {
                Qmsg.success("自动填入访问码");
                accessInputElement.value = netDiskInfo.accessCode;
                accessInputElement._value = netDiskInfo.accessCode;
                utils.dispatchEvent(accessInputElement, "input");
                utils.dispatchEvent(
                  document.querySelector("div.button"),
                  "click"
                );
              });
            });
        }
      },
      /**
       * 中国移动云盘
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      hecaiyun(netDiskInfo) {
        if (window.location.hostname === "caiyun.139.com") {
          /* 桌面端 */
          log.success(["自动填写链接", netDiskInfo]);
          utils.waitNode("#token-input").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.dispatchEvent(element, "input");
            document
              .querySelector("#homepage div.token div.token-form a")
              .click();
          });
          /* 移动端 */
          utils
            .waitNode("#app div.token-form input[type=text]")
            .then((element) => {
              if (!utils.isVisible(element)) {
                log.error("输入框不可见,不输入密码");
                return;
              }
              Qmsg.success("自动填入访问码");
              element.value = netDiskInfo.accessCode;
              utils.dispatchEvent(element, "input");
              document.querySelector("div.token-form a.btn-token").click();
            });
        }
      },
      /**
       * 阿里云盘
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      aliyun(netDiskInfo) {
        if (
          window.location.hostname === "www.aliyundrive.com" ||
          window.location.hostname === "www.alipan.com"
        ) {
          /* 桌面端 */
          log.success(["自动填写链接", netDiskInfo]);
          utils.waitNode("#root input.ant-input").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.getReactObj(element).reactFiber.memoizedProps.onChange({
              currentTarget: element,
              target: element,
            });
            document.querySelector('#root button[type="submit"]').click();
          });

          /* ------------------------------------ */

          /* 移动端 */
          utils.waitNode("#root input[name=pwd]").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.getReactObj(element).reactFiber.memoizedProps.onChange({
              currentTarget: element,
              target: element,
            });
            document.querySelector('#root button[type="submit"]').click();
          });
        }
      },
      /**
       * 文叔叔
       * 暂时没找到有密码的链接
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      wenshushu(netDiskInfo) {},
      /**
       * 奶牛
       * 暂时没找到有密码的链接
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      nainiu(netDiskInfo) {},
      /**
       * 123云盘
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      _123pan(netDiskInfo) {
        if (window.location.hostname === "www.123pan.com") {
          /* 桌面端 */
          log.success(["自动填写链接", netDiskInfo]);
          utils
            .waitNode("#app .ca-fot input.ant-input[type=text]")
            .then((element) => {
              if (!utils.isVisible(element)) {
                log.error("输入框不可见,不输入密码");
                return;
              }
              Qmsg.success("自动填入访问码");
              utils.getReactObj(element).reactProps.onChange({
                target: {
                  value: netDiskInfo.accessCode,
                },
              });
              element.nextSibling.click();
            });

          /* ------------------------------------ */

          /* 移动端 */
          utils
            .waitNode("#app .appinput input.ant-input[type=text]")
            .then((element) => {
              if (!utils.isVisible(element)) {
                log.error("输入框不可见,不输入密码");
                return;
              }
              Qmsg.success("自动填入访问码");
              utils.getReactObj(element).reactProps.onChange({
                target: {
                  value: netDiskInfo.accessCode,
                },
              });
              element.nextSibling.click();
            });
        }
      },
      /**
       * 腾讯微云
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      weiyun(netDiskInfo) {
        if (window.location.hostname === "share.weiyun.com") {
          /* 桌面端 */
          log.success(["自动填写链接", netDiskInfo]);
          utils.waitNode("#app input.input-txt").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.dispatchEvent(element, "input");
            utils.dispatchEvent(element, "change");
            setTimeout(() => {
              document.querySelector(".form-item button.btn-main").click();
            }, 500);
          });

          /* ------------------------------------ */

          /* 移动端 */
          utils.waitNode(".input-wrap input.pw-input").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.dispatchEvent(element, "input");
            utils.dispatchEvent(element, "change");
            setTimeout(() => {
              document.querySelector(".pw-btn-wrap button.btn").click();
            }, 500);
          });
        }
      },
      /**
       * 迅雷
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      xunlei(netDiskInfo) {
        if (window.location.hostname === "pan.xunlei.com") {
          /* 桌面端 */
          log.success(["自动填写链接", netDiskInfo]);
          utils
            .waitNode("#__layout div.pass-input-wrap input.td-input__inner")
            .then((element) => {
              if (!utils.isVisible(element)) {
                log.error("输入框不可见,不输入密码");
                return;
              }
              log.error("输入框不可见,不输入密码");
              element.value = netDiskInfo.accessCode;
              utils.dispatchEvent(element, "input");
              utils.dispatchEvent(element, "change");
              setTimeout(() => {
                document
                  .querySelector(
                    "#__layout div.pass-input-wrap button.td-button"
                  )
                  .click();
              }, 500);
            });

          /* ------------------------------------ */

          /* 移动端 */
          utils
            .waitNode("#__layout div.pass-wrapper input.td-input__inner")
            .then((element) => {
              if (!utils.isVisible(element)) {
                log.error("输入框不可见,不输入密码");
                return;
              }
              log.error("输入框不可见,不输入密码");
              element.value = netDiskInfo.accessCode;
              utils.dispatchEvent(element, "input");
              utils.dispatchEvent(element, "change");
              setTimeout(() => {
                document
                  .querySelector("#__layout div.pass-wrapper button.td-button")
                  .click();
              }, 500);
            });
        }
      },
      /**
       * 115网盘
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      _115pan(netDiskInfo) {
        if (window.location.hostname === "115.com") {
          log.success(["自动填写链接", netDiskInfo]);
          utils.waitNode("input.text").then((element) => {
            if (!utils.isVisible(element)) {
              log.error("输入框不可见,不输入密码");
              return;
            }
            Qmsg.success("自动填入访问码");
            element.value = netDiskInfo.accessCode;
            utils.dispatchEvent(element, "input");
            document
              .querySelector("#js-share_code div.form-decode div.submit a")
              .click();
          });
        }
      },
      /**
       * 城通网盘
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      chengtong(netDiskInfo) {
        log.success(["自动填写链接", netDiskInfo]);
        utils.waitNode("#passcode").then((element) => {
          if (!utils.isVisible(element)) {
            log.error("输入框不可见,不输入密码");
            return;
          }
          Qmsg.success("自动填入访问码");
          element.value = netDiskInfo.accessCode;
          utils.dispatchEvent(element, "input");
          document
            .querySelector("#main-content .form-group button.btn[type=button]")
            .click();
        });
      },
      /**
       * 夸克网盘
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      kuake(netDiskInfo) {
        if (window.location.hostname === "pan.quark.cn") {
          log.success(["自动填写链接", netDiskInfo]);
          utils
            .waitNode("#ice-container input.ant-input[class*=ShareReceive]")
            .then((element) => {
              if (!utils.isVisible(element)) {
                log.error("输入框不可见,不输入密码");
                return;
              }
              Qmsg.success("自动填入访问码");
              (
                utils.getReactObj(element).reactProps ||
                utils.getReactObj(element).reactEventHandlers
              ).onChange({
                target: {
                  value: netDiskInfo.accessCode,
                },
              });
            });
        }
      },
      /**
       * 坚果云
       * 暂时没找到有密码的链接
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      jianguoyun(netDiskInfo) {},
      /**
       * OneDrive
       * 暂时没找到有密码的链接
       * @param {NetDiskAutoFillAccessCodeOption} netDiskInfo
       */
      onedrive(netDiskInfo) {},
    },
    /**
     *
     * @param {NetDiskAutoFillAccessCodeOption} value
     */
    setValue(value) {
      GM_setValue(this.key, value);
    },
    getValue() {
      return GM_getValue(this.key);
    },
  };

  /** Woker */
  const NetDiskWorker = {
    /** worker的Blob链接 */
    blobUrl: "",
    /** @type {Worker} */
    GM_matchWorker: void 0,
    /** 初始化Worker的Blob链接 */
    initWorkerBlobLink() {
      const handleMatch = `
			(() => {
        function ${NetDiskWorker.handleRegularMatch.toString()}

        function ${NetDiskWorker.uniqueArr}
        
        this.addEventListener(
          "message",
          function (event) {
            const data = event.data;
            let matchedList = [];
            ${NetDiskWorker.handleRegularMatch.name}(data,(matchData)=>{
              matchedList.push(matchData);
            })
            matchedList = ${NetDiskWorker.uniqueArr.name}(matchedList);
            this.postMessage({
              options: data,
              msg: "Match End",
              data: matchedList,
              startTime: data.startTime,
              endTime: Date.now(),
            });
          },
          {
            capture: true,
          }
        );
      })();
      `;
      let blob = new Blob([handleMatch]);
      NetDiskWorker.blobUrl = window.URL.createObjectURL(blob);
      log.info(`Worker Blob Link ===> ${NetDiskWorker.blobUrl}`);
    },
    /**
     * 处理规则匹配
     * @param {NetDiskWorkerOptions} data 数据
     * @param {( matchData: NetDiskWorkerMatchOption )=>void} callback 成功匹配到的回调
     */
    handleRegularMatch(data, callback) {
      const NetDiskRegularNameList = Object.keys(data.regular);
      for (const netDiskName of NetDiskRegularNameList) {
        const netDiskRegular = data.regular[netDiskName];
        for (let index = 0; index < netDiskRegular.length; index++) {
          const netDiskRegularItem = netDiskRegular[index];
          if (
            netDiskRegularItem["enable"] != null &&
            !netDiskRegularItem["enable"]
          ) {
            continue;
          }
          let regexpList = [];
          if (data.matchTextRange === "all") {
            regexpList.push(
              new RegExp(netDiskRegularItem["link_innerText"], "gi")
            );
            regexpList.push(
              new RegExp(netDiskRegularItem["link_innerHTML"], "gi")
            );
          } else if (data.matchTextRange === "innertext") {
            regexpList.push(
              new RegExp(netDiskRegularItem["link_innerText"], "gi")
            );
          } else if (data.matchTextRange === "innerhtml") {
            regexpList.push(
              new RegExp(netDiskRegularItem["link_innerHTML"], "gi")
            );
          } else {
            throw new TypeError("未知的匹配范围: " + data.matchTextRange);
          }
          for (const regexp of regexpList) {
            for (const text of data.textList) {
              let matchData = text.match(regexp);
              if (matchData && matchData.length) {
                callback({
                  netDiskName: netDiskName,
                  netDiskIndex: index,
                  data: matchData,
                });
              }
            }
          }
        }
      }
    },
    /**
     * 数组去重
     */
    uniqueArr(arr) {
      return arr.filter((obj, index, selfArray) => {
        return (
          index ===
          selfArray.findIndex((item) => {
            return JSON.stringify(item) === JSON.stringify(obj);
          })
        );
      });
    },
    /**
     * 初始化Worker对象
     */
    initWorker() {
      try {
        NetDiskWorker.GM_matchWorker = new Worker(NetDiskWorker.blobUrl);
        NetDiskWorker.GM_matchWorker.onmessage = NetDiskWorker.onMessage;
        NetDiskWorker.GM_matchWorker.onerror = NetDiskWorker.onError;
      } catch (error) {
        log.error([
          "初始化Worker失败,可能页面使用了Content-Security-Policy策略,使用另类方法",
          error.message,
        ]);
        NetDiskWorker.GM_matchWorker = {
          /**
           *
           * @param {NetDiskWorkerOptions} data
           */
          postMessage(data) {
            return new Promise((resolve, reject) => {
              /** @type {NetDiskWorkerMatchOption[]} */
              let matchedList = [];
              try {
                NetDiskWorker.handleRegularMatch(data, (matchData) => {
                  matchedList.push(matchData);
                });
              } catch (error) {
                NetDiskWorker.onError(error);
              } finally {
                matchedList = NetDiskWorker.uniqueArr(matchedList);
                NetDiskWorker.onMessage(
                  new MessageEvent("message", {
                    data: {
                      options: data,
                      msg: "Match End",
                      data: matchedList,
                      startTime: data.startTime,
                      endTime: Date.now(),
                    },
                  })
                );
                resolve();
              }
            });
          },
        };
      }
    },
    /**
     * 传递数据给worker内进行处理匹配
     * @param { NetDiskWorkerOptions } message 数据
     * @param { ?StructuredSerializeOptions } options 配置
     */
    postMessage(message, options) {
      NetDiskWorker.GM_matchWorker.postMessage(message, options);
    },
    /**
     * Worker的onmessage
     * @param {MessageEvent<NetDiskWorkerCallBackOptions>} event
     */
    onMessage(event) {
      const data = event.data;
      if (data.data.length) {
        log.success(
          `成功匹配${data.data.length}个,用时${Date.now() - data.startTime}ms`
        );
      }
      if (event.data.options.from === "PasteText") {
        NetDiskUI.matchPasteText.matchEndCallBack(data);
      }
      NetDiskWorker.successCallBack(data);
    },
    /**
     * Worker的onerror
     * @param {ErrorEvent} error
     */
    onError(error) {
      NetDiskWorker.errorCallBack(error);
    },
    /**
     * worker处理文件匹配后的回调
     * @param {NetDiskWorkerCallBackOptions} options
     */
    successCallBack(options) {
      /* 匹配为空,释放锁 */
      if (!options.data.length) {
        NetDiskWorker.matchingEndCallBack();
        return;
      }
      /** @type {NetiDiskHandleObject[]} */
      const handleNetDiskList = [];
      for (const matchData of options.data) {
        /* 已匹配到的网盘,用于显示图标 */
        NetDisk.matchLink.add(matchData.netDiskName);
        /**
         * 匹配到的可能很多,使用集合去重
         * @type {Set<string>}
         */
        let matchLinkSet = new Set();
        matchData.data.forEach((item) => {
          matchLinkSet.add(item);
        });

        matchLinkSet.forEach((item) => {
          let handleLink = NetDisk.handleLink(
            matchData.netDiskName,
            matchData.netDiskIndex,
            item
          );
          if (handleLink) {
            handleNetDiskList.push({
              shareCode: handleLink.shareCode,
              accessCode: handleLink.accessCode,
              netDiskName: matchData.netDiskName,
              netDiskIndex: matchData.netDiskIndex,
              matchText: item,
            });
          }
        });
      }
      /* 过滤掉重复的 */
      let filterHandleNetDiskList = handleNetDiskList.filter(
        (value, index, selfArray) => {
          let isFind =
            selfArray.findIndex((obj) => {
              /* 过滤掉同样配置的 */
              /* 或者是netDiskName、netDiskIndex相同,且shareCode前面存在重复的 */
              return (
                //JSON.stringify(obj) === JSON.stringify(value)
                obj.accessCode === value.accessCode &&
                obj.netDiskIndex === value.netDiskIndex &&
                obj.netDiskName === value.netDiskName &&
                obj.shareCode === value.shareCode
                //(obj.netDiskName === value.netDiskName &&
                //  obj.netDiskIndex === value.netDiskIndex &&
                //  obj.shareCode.startsWith(value.shareCode)) ||
                //value.shareCode.startsWith(obj.shareCode)
              );
            }) === index;
          return isFind;
        }
      );
      filterHandleNetDiskList.forEach((item) => {
        if (NetDisk.tempLinkDict.has(item.netDiskName)) {
          let currentTempDict = NetDisk.tempLinkDict.get(item.netDiskName);
          currentTempDict.set(item.shareCode, item);
        }
      });
      filterHandleNetDiskList.forEach((item) => {
        const { shareCode, accessCode, netDiskName, netDiskIndex, matchText } =
          item;
        const currentRegular = NetDisk.regular[netDiskName][netDiskIndex];
        if (
          currentRegular.shareCodeExcludeRegular &&
          Array.isArray(currentRegular.shareCodeExcludeRegular)
        ) {
          /* 排除掉在目标规则已匹配到的shareCode */
          for (const excludeRegularName of currentRegular.shareCodeExcludeRegular) {
            let excludeDict = NetDisk.linkDict.get(excludeRegularName);
            let currentTempDict = NetDisk.tempLinkDict.get(excludeRegularName);
            if (
              excludeDict.startsWith(shareCode) ||
              currentTempDict.startsWith(shareCode)
            ) {
              log.warn(
                `${netDiskName}:该分享码【${shareCode}】与已匹配到该分享码的规则【${excludeRegularName}】冲突`
              );
              return;
            }
          }
        }

        /** 当前存储的 */
        const currentDict = NetDisk.linkDict.get(netDiskName);
        NetDisk.hasMatchLink = true;
        if (currentDict.startsWith(shareCode)) {
          /* 存在该访问码 */
          /* 根据shareCode获取accessCode和netDiskIndex信息 */
          let shareCodeDict = currentDict.getStartsWith(shareCode);
          if (
            typeof shareCodeDict.isForceAccessCode === "boolean" &&
            shareCodeDict.isForceAccessCode
          ) {
            /* 该访问码已被锁定,禁止修改 */
            return;
          }
          if (utils.isNotNull(shareCodeDict.accessCode)) {
            /* 已匹配到的访问码不为空,不设置新的 */
            return;
          }
          if (utils.isNull(accessCode)) {
            /* 访问码为空,不设置新的 */
            return;
          }

          currentDict.set(
            shareCode,
            NetDisk.getLinkDickObj(accessCode, netDiskIndex, false, matchText)
          );
          NetDiskUI.view.changeLinkView(
            netDiskName,
            netDiskIndex,
            shareCode,
            accessCode,
            matchText
          );
          log.info(
            `该匹配项无密码,设置密码 ${netDiskName} ${netDiskIndex}: ${shareCode}  ===> ${accessCode}`
          );
        } else {
          /* 不存在该访问码,添加新的进去 */
          currentDict.set(
            shareCode,
            NetDisk.getLinkDickObj(accessCode, netDiskIndex, false, matchText)
          );
          NetDiskUI.isMatchedNetDiskIconMap.add(netDiskName);
          NetDiskUI.view.addLinkView(
            netDiskName,
            netDiskIndex,
            shareCode,
            accessCode,
            matchText
          );
          log.success(
            `添加链接 ${netDiskName} ${netDiskIndex}: ${shareCode}  ===> ${accessCode}`
          );
        }
      });

      /* 清空临时的 */
      Object.keys(NetDisk.tempLinkDict.getItems()).forEach((keyName) => {
        NetDisk.tempLinkDict.get(keyName).clear();
      });
      if (NetDisk.hasMatchLink) {
        switch (NetDiskUIMenuData.netdiskBehaviorMode) {
          case "suspension_smallwindow".toLowerCase():
            if (
              GM_getValue(
                "current_suspension_smallwindow_mode",
                "suspension"
              ) === "suspension"
            ) {
              NetDiskUI.suspension.show();
            } else {
              NetDiskUI.view.show();
            }
            break;
          case "suspension_window".toLowerCase():
            NetDiskUI.suspension.show();
            break;
          case "smallwindow".toLowerCase():
            NetDiskUI.view.show();
            break;
          default:
            log.error([
              "未知的行为模式:" + NetDiskUIMenuData.netdiskBehaviorMode,
            ]);
        }
      }
      NetDiskWorker.matchingEndCallBack();
    },
    /**
     * Worker失败回调
     * @param {object} error
     */
    errorCallBack(error) {
      NetDiskWorker.matchingEndCallBack(true);
      log.error(["Worker Error", error]);
    },
    /**
     * 匹配结束回调
     * @param {boolean} isNow 是否立刻释放锁
     */
    matchingEndCallBack(isNow) {
      if (isNow) {
        NetDiskUI.isHandleMatch = false;
      } else {
        const delaytime =
          parseFloat(
            GM_getValue("delaytime", NetDiskData.defaultMatchIntervalTime)
          ) * 1000;
        setTimeout(() => {
          NetDiskWorker.matchingEndCallBack(true);
        }, delaytime);
      }
    },
  };

  /** 网盘-引用/获取文件 */
  const NetDiskRequire = {
    requiredFileMap: new Map(),
    /**
     * 网络加载文件
     * @param {string} path
     * @param {HttpxDetails} options
     * @returns
     */
    async file(path, options) {
      if (utils.isNull(path)) {
        log.error(["NetDiskRequire.file的参数path为空", path]);
        return false;
      }
      if (this.requiredFileMap.has(path)) {
        log.warn(["NetDiskRequire.file的参数path已引入过", path]);
        return true;
      }
      let getResp = await httpx.get(path, options);
      if (!getResp.status) {
        return false;
      }
      let jsText = getResp.data.responseText;
      this.requiredFileMap.set(path, jsText);
      log.info(["加载js文件", path]);
      unsafeWindow.eval(jsText);
      await utils.sleep(300);
    },
  };

  /** 网盘-自定义规则 */
  const NetDiskCustomRules = {
    key: "userRule",
    dataKey: "userRuleData",
    $data: {
      /**
       * @type {UtilsDictionaryConstructor<string,NetDiskSettingMenuDetails>}
       */
      settingViewDetails: new utils.Dictionary(),
    },
    init() {
      if (typeof GM_getValue(this.dataKey) !== "object") {
        GM_setValue(this.dataKey, {});
      }
      let userRule = this.parseRule(this.getRule());
      for (let ruleName in userRule) {
        if (ruleName in NetDisk.regular) {
          /* 如果规则已存在(已内置),自定义规则先放在前面匹配 */
          NetDisk.regular[ruleName] = [
            ...userRule[ruleName],
            ...NetDisk.regular[ruleName],
          ];
        } else {
          NetDisk.regular[ruleName] = userRule[ruleName];
        }
      }
    },
    getCSS() {
      return `
      .pops[type-value=confirm] .pops-confirm-content{
        overflow: hidden;
      }
      /* textarea美化 */
      .pops.whitesevPopNetDiskCustomRules[type-value="confirm"] .pops-confirm-content textarea{
        width: 100%;
        height: 100%;
        border: none;
        outline: none;
        padding: 0;
        margin: 0;
        -webkit-appearance: none;
        -moz-appearance: none;
        appearance: none;
        background-image: none;
        background-color: transparent;

        display: inline-block;
        resize: vertical;
        padding: 5px 15px;
        line-height: 1.5;
        box-sizing: border-box;
        border: 1px solid #dcdfe6;
        transition: border-color 0.2s cubic-bezier(0.645, 0.045, 0.355, 1);
        appearance: none;
        resize: none;
      }
      /* 获得焦点 */
      .pops.whitesevPopNetDiskCustomRules[type-value="confirm"] .pops-confirm-content textarea:focus{
        outline: none;
        border-color: #3677f0;
      }
      /* 提示文字 */
      .pops.whitesevPopNetDiskCustomRules[type-value="confirm"] .pops-confirm-content textarea::placeholder {
        color: #c0c4cc;
      }
      /* 鼠标hover */
      .pops.whitesevPopNetDiskCustomRules[type-value="confirm"] .pops-confirm-content textarea:hover {
        border-color: #c0c4cc;
      }
      `;
    },
    /**
     * 添加/编辑规则
     * @param {boolean} isEdit
     * @param {?string} ruleKey 当isEdit为true时,传入该值
     */
    showUI(isEdit, ruleKey) {
      let titleText = "添加";
      if (isEdit) {
        titleText = "编辑";
      }
      titleText += "自定义规则";
      /** @type {HTMLTextAreaElement} */
      let $ruleInput = null;

      /**
       * 保存按钮的回调
       * @param {PopsBtnCallBackEvent} event
       */
      function saveCallBack(event) {
        /**
         * 检测regexp项是否符合规定
         * @param {NetDiskUserCustomRuleRegexp} ruleRegExp
         * @returns
         */
        function checkRegExp(ruleRegExp) {
          if (typeof ruleRegExp["link_innerText"] !== "string") {
            Qmsg.error("regexp缺失的键名: link_innerText,类型: string");
            return;
          }
          if (typeof ruleRegExp["link_innerHTML"] !== "string") {
            Qmsg.error("regexp缺失的键名: link_innerHTML,类型: string");
            return;
          }
          if (typeof ruleRegExp["shareCode"] !== "string") {
            Qmsg.error("regexp缺失的键名: shareCode,类型: string");
            return;
          }
          if (typeof ruleRegExp["shareCodeNeedRemoveStr"] !== "string") {
            Qmsg.error(
              "regexp缺失的键名: shareCodeNeedRemoveStr,类型: string"
            );
            return;
          }
          if (typeof ruleRegExp["uiLinkShow"] !== "string") {
            Qmsg.error("regexp缺失的键名: uiLinkShow,类型: string");
            return;
          }
          if (typeof ruleRegExp["blank"] !== "string") {
            Qmsg.error("regexp缺失的键名: blank,类型: string");
            return;
          }
          if (typeof ruleRegExp["copyUrl"] !== "string") {
            Qmsg.error("regexp缺失的键名: copyUrl,类型: string");
            return;
          }
          if (
            typeof ruleRegExp["accessCode"] === "string" &&
            typeof ruleRegExp["checkAccessCode"] !== "string"
          ) {
            Qmsg.error("regexp设置了accessCode但是没有设置checkAccessCode");
            return;
          }
          if (
            typeof ruleRegExp["accessCode"] !== "string" &&
            typeof ruleRegExp["checkAccessCode"] === "string"
          ) {
            Qmsg.error("regexp设置了checkAccessCode但是没有设置accessCode");
            return;
          }
          return true;
        }
        /**
         * 检测设置项是否符合规定
         * @param {NetDiskUserCustomRuleSetting} ruleSetting
         */
        function checkSetting(ruleSetting) {
          if (typeof ruleSetting["name"] !== "string") {
            Qmsg.error("setting缺失的键名: name,类型: string");
            return;
          }
          if (typeof ruleSetting["enable"] !== "boolean") {
            Qmsg.error("setting缺失的键名: enable,类型: boolean");
            return;
          }
          return true;
        }
        try {
          /**
           * @type {NetDiskUserCustomRule}
           */
          let ruleJSON = JSON.parse($ruleInput.value);
          if (typeof ruleJSON !== "object") {
            Qmsg.error("该规则不是object类型");
            return;
          }
          if (typeof ruleJSON["key"] !== "string") {
            Qmsg.error("缺失的键名: key,类型: string");
            return;
          }
          if (typeof ruleJSON["regexp"] !== "object") {
            Qmsg.error("缺失的键名: regexp,类型: object|Arrany");
            return;
          }
          if (typeof ruleJSON["setting"] !== "object") {
            Qmsg.error("缺失的键名: setting,类型: object");
            return;
          }
          if (Array.isArray(ruleJSON["regexp"])) {
            for (const regexpItem of ruleJSON["regexp"]) {
              if (!checkRegExp(regexpItem)) {
                return;
              }
            }
          } else {
            if (!checkRegExp(ruleJSON["regexp"])) {
              return;
            }
          }

          if (!checkSetting(ruleJSON["setting"])) {
            return;
          }
          if (isEdit) {
            NetDiskCustomRules.setRule(ruleKey, ruleJSON);
          } else {
            NetDiskCustomRules.addRule(ruleJSON);
          }
          Qmsg.success("保存成功");
          return ruleJSON;
        } catch (error) {
          log.error(error);
          Qmsg.error(error.message, {
            html: true,
            timeout: 3500,
          });
          return;
        }
      }
      /**
       * 调试按钮的回调
       * @param {PopsBtnCallBackEvent} event
       */
      function debugCallBack(event) {
        let ruleJSON = saveCallBack(event);
        if (!ruleJSON) {
          return;
        }
        NetDiskCustomRules.showDebugUI(ruleJSON);
      }
      /**
       * 格式化按钮的回调
       * @param {PopsBtnCallBackEvent} event
       */
      function formatCallBack(event) {
        try {
          let ruleJSON = JSON.parse($ruleInput.value);
          let ruleJSONString = NetDiskCustomRules.getFormatRule(ruleJSON);
          $ruleInput.value = ruleJSONString;
          Qmsg.success("格式化成功");
        } catch (error) {
          log.error(error);
          Qmsg.error(error.message, {
            html: true,
            timeout: 3500,
          });
        }
      }
      let dialog = NetDiskPops.confirm(
        {
          title: {
            text: titleText,
            position: "center",
          },
          content: {
            text: `<textarea class="netdisk-custom-rules" placeholder="请输入自定义规则"></textarea>`,
            html: true,
          },
          btn: {
            merge: true,
            mergeReverse: false,
            reverse: false,
            position: "space-between",
            ok: {
              text: "保存",
              callback: saveCallBack,
            },
            cancel: {
              text: "调试",
              callback: debugCallBack,
            },
            other: {
              enable: true,
              text: "格式化",
              type: "xiaomi-primary",
              callback: formatCallBack,
            },
          },
          class: "whitesevPopNetDiskCustomRules",
          style: this.getCSS(),
        },
        NetDiskUI.popsStyle.customRulesView
      );
      $ruleInput = dialog.$shadowRoot.querySelector("textarea");
      if (isEdit) {
        let rule = this.getRule(ruleKey);
        $ruleInput.value = this.getFormatRule(rule);
      } else {
        $ruleInput.value = this.getTemplateRule();
      }
    },
    /**
     * 调试规则
     * @param {NetDiskUserCustomRule} ruleJSON
     */
    showDebugUI(ruleJSON) {
      if (utils.isNull(ruleJSON.regexp)) {
        Qmsg.error("请先配置regexp");
        return;
      }
      let that = this;
      let customRule = this.parseRule([ruleJSON]);
      let regexp = customRule[ruleJSON.key];
      let dialog = NetDiskPops.confirm(
        {
          title: {
            text: `调试规则 ${ruleJSON.key}`,
            position: "center",
          },
          content: {
            text: `
          <div class="custom-rule-container">
            <textarea class="custom-rule-match-text" placeholder="请输入需要测试匹配的字符串"></textarea>
            <div class="custom-rule-input-container">
              <select class="custom-rule-select-regexp"></select>
              <button class="custom-rule-run-match-button" type="primary" data-icon="" data-righticon="false">
                <span>执行</span>
              </button>
            </div>
          </div>
          <div class="custom-rule-match-log">
            <div>匹配日志↓</div>
            <div class="custom-rule-match-log-container"></div>
          </div>
          `,
            html: true,
          },
          btn: {
            ok: {
              enable: false,
            },
          },
          style: `
        .custom-rule-container{
          display: flex;
          align-items: center;
        }
        .custom-rule-select-regexp{
          width: 100%;
          height: 32px;
          line-height: 32px;
          border: 1px solid rgb(184, 184, 184, var(--pops-bd-opacity));
          border-radius: 5px;
          text-align: center;
          outline: 0;
          background: rgb(255, 255, 255, var(--pops-bg-opacity));
          box-shadow: none;
        }
        .custom-rule-input-container{
          display: flex;
          flex-wrap: wrap;
          justify-content: center;
          margin: 5px;
          width: 30%;
        }
        .custom-rule-select-regexp-item{

        }
        button.custom-rule-run-match-button{
          margin-top: 5px;
        }
        textarea.custom-rule-match-text{
          width: 100%;
          min-height: 70px;
          outline: none;
          margin: 0px;
          background-image: none;
          background-color: transparent;
          display: inline-block;
          resize: vertical;
          padding: 5px;
          line-height: 1.5;
          box-sizing: border-box;
          border: 1px solid rgb(220, 223, 230);
          appearance: none;
        }
        .custom-rule-match-log{

        }
        .custom-rule-match-log-container{
          padding: 5px;
          background: rgb(229, 229, 229);
        }
        .custom-rule-match-log-container p{
          margin: 2px 0px;
          border-bottom: 1px solid #000000;
        }
        .custom-rule-match-log-container p:last-child{
          border-bottom: 0px;
          margin-bottom: 0px;
        }
        .custom-rule-match-log-container p[data-tag]{
          
        }
        .custom-rule-match-log-container p[data-tag="info"]{

        }
        .custom-rule-match-log-container p[data-tag="success"]{
          color: green;
        }
        .custom-rule-match-log-container p[data-tag="warn"]{
          color: yellow;
        }
        .custom-rule-match-log-container p[data-tag="error"]{
          color: red;
        }
        `,
        },
        NetDiskUI.popsStyle.customRuleDebugView
      );
      /** @type {HTMLSelectElement} */
      let $select = dialog.$shadowRoot.querySelector(
        ".custom-rule-select-regexp"
      );
      /** @type {HTMLTextAreaElement} */
      let $matchText = dialog.$shadowRoot.querySelector(
        ".custom-rule-match-text"
      );
      /** @type {HTMLDivElement} */
      let $log = dialog.$shadowRoot.querySelector(
        ".custom-rule-match-log-container"
      );
      /** @type {HTMLButtonElement} */
      let $button = dialog.$shadowRoot.querySelector(
        ".custom-rule-run-match-button"
      );
      regexp.forEach((regExpItem, index) => {
        $select.appendChild(
          DOMUtils.createElement("option", {
            className: "custom-rule-select-regexp-item",
            innerText: "regexp下标:" + index,
            "data-value": regExpItem,
          })
        );
      });
      /**
       * 设置日志输出
       * @param {"info"|"error"|"success"|"warn"} tag 日志等级
       * @param {...any[]} args
       */
      function setLog(tag, ...args) {
        let text = "";
        args.forEach((item) => {
          if (text !== "") {
            text += "\n";
          }
          if (typeof item !== "string") {
            text += JSON.stringify(item, void 0, 4);
          } else {
            text += item;
          }
        });
        let logElement = DOMUtils.createElement(
          "p",
          {
            innerText: text,
          },
          {
            "data-tag": tag,
          }
        );
        DOMUtils.append($log, logElement);
      }
      /**
       * 清空日志
       */
      function clearLog() {
        $log.innerHTML = "";
      }
      /**
       * 日志回调
       * @param {NetDiskDebugLogData} logData
       */
      function logCallBack(logData) {
        if (Array.isArray(logData.msg)) {
          setLog(logData.status ? "info" : "error", ...logData.msg);
        } else {
          setLog(logData.status ? "info" : "error", logData.msg);
        }
        if (!logData.status) {
          setLog("error", "执行完毕");
        }
      }
      function debugRunClickEvent() {
        if (utils.isNull($matchText.value)) {
          Qmsg.error("请先输入待匹配的字符串");
          return;
        }
        /* 清空日志 */
        clearLog();
        let netDiskName = ruleJSON.key;
        let netDiskIndex = $select.selectedIndex;
        let selectRegExp = $select.selectedOptions[netDiskIndex]["data-value"];
        log.info(["当前选中的规则: ", selectRegExp]);
        let testCustomRule = {};
        testCustomRule[ruleJSON.key] = [selectRegExp];
        /** @type {string[]} */
        let matchTextList = [];
        NetDiskWorker.handleRegularMatch(
          {
            regular: testCustomRule,
            textList: [$matchText.value],
            matchTextRange: "all",
          },
          (matchData) => {
            matchTextList.push(...matchData.data);
          }
        );
        if (!matchTextList.length) {
          setLog("error", "未成功匹配到数据");
          return;
        }
        matchTextList = NetDiskWorker.uniqueArr(matchTextList);
        setLog("info", "成功匹配到的数据 ==> ", matchTextList);
        matchTextList.forEach((matchText, index) => {
          setLog("success", "当前处理的字符串: " + matchText);
          setLog("success", "当前执行: 对shareCode进行处理获取");
          let shareCode = NetDiskDebug.handleShareCode(
            matchText,
            selectRegExp,
            logCallBack
          );
          if (utils.isNull(shareCode)) {
            return;
          }
          setLog("info", " ");
          setLog("info", `================分割线================`);
          setLog("info", " ");
          setLog("success", "当前执行: 对accessCode进行处理获取");
          let accessCode = NetDiskDebug.handleAccessCode(
            matchText,
            selectRegExp,
            logCallBack
          );
          setLog("info", " ");
          setLog("info", `================分割线================`);
          setLog("info", " ");
          setLog("success", "当前执行: 对uiLinkShow进行处理获取");
          let uiLinkShow = NetDiskDebug.handleLinkShow(
            matchText,
            selectRegExp,
            shareCode,
            accessCode,
            logCallBack
          );
          setLog("info", " ");
          setLog("info", `================分割线================`);
          setLog("info", " ");
          setLog("success", "当前执行: 对blank进行处理获取");
          let blankUrl = NetDiskDebug.handleBlank(
            matchText,
            selectRegExp,
            shareCode,
            accessCode,
            logCallBack
          );
          setLog("info", " ");
          setLog("info", `================分割线================`);
          setLog("info", " ");
          setLog("success", "当前执行: 对copyUrl进行处理获取");
          let copyUrl = NetDiskDebug.handleCopyUrl(
            matchText,
            selectRegExp,
            shareCode,
            accessCode,
            logCallBack
          );
          setLog("success", "执行完毕");
        });
      }
      DOMUtils.on($button, "click", void 0, debugRunClickEvent);
    },
    /**
     * 上下文环境
     * @param {NetDiskUserCustomRule} rule
     */
    getBindContext(rule) {
      function setValue(key, value) {
        let localData = GM_getValue(NetDiskCustomRules.dataKey);
        let ruleData = localData[rule.key];
        ruleData[key] = value;
        GM_setValue(NetDiskCustomRules.dataKey, localData);
      }
      function getValue(key, defaultValue) {
        let localData = GM_getValue(NetDiskCustomRules.dataKey);
        let ruleData = localData[rule.key];
        return ruleData[key] ?? defaultValue;
      }
      function deleteValue(key) {
        let localData = GM_getValue(NetDiskCustomRules.dataKey);
        let ruleData = localData[rule.key];
        Reflect.deleteProperty(ruleData, key);
        GM_setValue(NetDiskCustomRules.dataKey, localData);
      }
      return {
        rule: rule,
        NetDiskRequire: NetDiskRequire,
        CryptoJS: Cryptojs,
        httpx: httpx,
        utils: utils,
        DOMUtils: DOMUtils,
        window: window,
        unsafeWindow: unsafeWindow,
        NetDiskCheckLinkValidity: NetDiskCheckLinkValidity,
        log: log,
        Qmsg: Qmsg,
        pops: pops,
        setValue: setValue,
        getValue: getValue,
        deleteValue: deleteValue,
      };
    },
    /**
     * 把用户自定义规则进行转换成脚本规则
     * @param {NetDiskUserCustomRule[]} localRule
     * @returns {NetDiskRegular}
     */
    parseRule(localRule) {
      /**
       *
       * @param {NetDiskUserCustomRuleRegexp} ruleRegExp
       */
      function parseRegExp(ruleRegExp) {
        if (typeof ruleRegExp["shareCode"] === "string") {
          ruleRegExp["shareCode"] = new RegExp(ruleRegExp["shareCode"], "ig");
        }
        if (typeof ruleRegExp["shareCodeNeedRemoveStr"] === "string") {
          ruleRegExp["shareCodeNeedRemoveStr"] = new RegExp(
            ruleRegExp["shareCodeNeedRemoveStr"],
            "ig"
          );
        }
        if (typeof ruleRegExp["shareCodeNotMatch"] === "string") {
          ruleRegExp["shareCodeNotMatch"] = new RegExp(
            ruleRegExp["shareCodeNotMatch"],
            "ig"
          );
        }
        if (typeof ruleRegExp["checkAccessCode"] === "string") {
          ruleRegExp["checkAccessCode"] = new RegExp(
            ruleRegExp["checkAccessCode"],
            "ig"
          );
        }
        if (typeof ruleRegExp["accessCode"] === "string") {
          ruleRegExp["accessCode"] = new RegExp(ruleRegExp["accessCode"], "ig");
        }
        if (typeof ruleRegExp["acceesCodeNotMatch"] === "string") {
          ruleRegExp["acceesCodeNotMatch"] = new RegExp(
            ruleRegExp["acceesCodeNotMatch"],
            "ig"
          );
        }
        if (typeof ruleRegExp["paramMatch"] === "string") {
          ruleRegExp["paramMatch"] = new RegExp(ruleRegExp["paramMatch"], "i");
        }
        return ruleRegExp;
      }
      let userRule = {};
      for (const userRuleItem of localRule) {
        let userRegExp = userRuleItem.regexp;
        if (userRuleItem.setting) {
          /** 设置 @type {NetDiskSettingMenuDetails} */
          let viewDetails = {
            type: userRuleItem.setting.name,
            key: userRuleItem.key,
            isUserRule: true,
          };
          let userRegExpOther = {};
          /* 是否启用,设置默认值 */
          this.initDefaultValue(
            `${userRuleItem.key}-enable`,
            Boolean(userRuleItem.setting["enable"])
          );

          viewDetails["enable"] = GM_getValue(`${userRuleItem.key}-enable`);

          userRegExpOther["enable"] = GM_getValue(`${userRuleItem.key}-enable`);

          if (typeof userRuleItem.setting["isBlank"] === "boolean") {
            /* 新标签页打开 */
            viewDetails["isBlank"] = true;
            this.initDefaultValue(
              `${userRuleItem.key}-open-enable`,
              Boolean(userRuleItem.setting["isBlank"])
            );
          }
          if (
            typeof userRuleItem.setting["openBlankWithCopyAccessCode"] ===
            "boolean"
          ) {
            /* 跳转时复制访问码 */
            viewDetails["openBlankWithCopyAccessCode"] = true;
            this.initDefaultValue(
              `${userRuleItem.key}-open-blank-with-copy-accesscode`,
              Boolean(userRuleItem.setting["openBlankWithCopyAccessCode"])
            );
          }
          if (typeof userRuleItem.setting["isForward"] === "boolean") {
            /* 直接进行scheme转发链接 */
            viewDetails["isForward"] = true;
            this.initDefaultValue(
              `${userRuleItem.key}-forward-scheme-enable`,
              Boolean(userRuleItem.setting["isForward"])
            );
          }
          if (typeof userRuleItem.setting["schemeUri"] === "boolean") {
            /* scheme转发的字符串格式 */
            viewDetails["schemeUri"] = true;
            this.initDefaultValue(`${userRuleItem.key}-static-scheme-uri`, "");
          }
          if (typeof userRuleItem.setting["checkLinkValidity"] === "boolean") {
            /* 用于验证链接有效性 */
            viewDetails["checkLinkValidity"] = true;
            this.initDefaultValue(
              `${userRuleItem.key}-check-link-valid`,
              Boolean(userRuleItem.setting["checkLinkValidity"])
            );
            userRegExpOther["checkLinkValidity"] = GM_getValue(
              `${userRuleItem.key}-check-link-valid`
            );
          }
          if (
            typeof userRuleItem.setting["innerTextAccessCodeBeforeMaxRange"] ===
            "number"
          ) {
            /* text-提取码间隔前的字符长度 */
            viewDetails["innerTextAccessCodeBeforeMaxRange"] =
              userRuleItem.setting["innerTextAccessCodeBeforeMaxRange"];
            this.initDefaultValue(
              `${userRuleItem.key}_innerText_accessCode_before_max_range`,
              viewDetails["innerTextAccessCodeBeforeMaxRange"]
            );
            userRegExpOther["innerTextAccessCodeBeforeMaxRange"] =
              NetDiskLocalData.innerTextAccessCodeBeforeMaxRange(
                userRuleItem.key
              );
          }
          if (
            typeof userRuleItem.setting["innerTextAccessCodeAfterMaxRange"] ===
            "number"
          ) {
            /* text-提取码间隔后的字符长度 */
            viewDetails["innerTextAccessCodeAfterMaxRange"] =
              userRuleItem.setting["innerTextAccessCodeAfterMaxRange"];
            this.initDefaultValue(
              `${userRuleItem.key}_innerText_accessCode_after_max_range`,
              viewDetails["innerTextAccessCodeAfterMaxRange"]
            );
            userRegExpOther["innerTextAccessCodeAfterMaxRange"] =
              NetDiskLocalData.innerTextAccessCodeAfterMaxRange(
                userRuleItem.key
              );
          }
          if (
            typeof userRuleItem.setting["innerHTMLAccessCodeBeforeMaxRange"] ===
            "number"
          ) {
            /* html-提取码间隔前的字符长度 */
            viewDetails["innerHTMLAccessCodeBeforeMaxRange"] =
              userRuleItem.setting["innerHTMLAccessCodeBeforeMaxRange"];
            this.initDefaultValue(
              `${userRuleItem.key}_innerHTML_accessCode_before_max_range`,
              viewDetails["innerHTMLAccessCodeBeforeMaxRange"]
            );
            userRegExpOther["innerHTMLAccessCodeBeforeMaxRange"] =
              NetDiskLocalData.innerHTMLAccessCodeBeforeMaxRange(
                userRuleItem.key
              );
          }
          if (
            typeof userRuleItem.setting["innerHTMLAccessCodeAfterMaxRange"] ===
            "number"
          ) {
            /* html-提取码间隔后的字符长度 */
            viewDetails["innerHTMLAccessCodeAfterMaxRange"] =
              userRuleItem.setting["innerHTMLAccessCodeAfterMaxRange"];
            this.initDefaultValue(
              `${userRuleItem.key}_innerHTML_accessCode_after_max_range`,
              viewDetails["innerHTMLAccessCodeAfterMaxRange"]
            );
            userRegExpOther["innerHTMLAccessCodeAfterMaxRange"] =
              NetDiskLocalData.innerHTMLAccessCodeAfterMaxRange(
                userRuleItem.key
              );
          }
          /**
           *
           * @param {NetDiskRegularOption} userRegexp
           */
          function handleRegExp(userRegexp) {
            if ("enable" in userRegExpOther) {
              userRegexp["enable"] = userRegExpOther["enable"];
            }
            if ("checkLinkValidity" in userRegExpOther) {
              userRegexp["checkLinkValidity"] =
                userRegExpOther["checkLinkValidity"];
            }
            if ("innerTextAccessCodeBeforeMaxRange" in userRegExpOther) {
              userRegexp["link_innerText"] = NetDisk.replaceParam(
                userRuleItem.key,
                {
                  innerTextAccessCodeBeforeMaxRange:
                    userRegExpOther["innerTextAccessCodeBeforeMaxRange"],
                }
              );
            }
            if ("innerTextAccessCodeAfterMaxRange" in userRegExpOther) {
              userRegexp["link_innerText"] = NetDisk.replaceParam(
                userRuleItem.key,
                {
                  innerTextAccessCodeAfterMaxRange:
                    userRegExpOther["innerTextAccessCodeAfterMaxRange"],
                }
              );
            }
            if ("innerHTMLAccessCodeBeforeMaxRange" in userRegExpOther) {
              userRegexp["link_innerHTML"] = NetDisk.replaceParam(
                userRuleItem.key,
                {
                  innerHTMLAccessCodeBeforeMaxRange:
                    userRegExpOther["innerHTMLAccessCodeBeforeMaxRange"],
                }
              );
            }
            if ("innerHTMLAccessCodeAfterMaxRange" in userRegExpOther) {
              userRegexp["link_innerHTML"] = NetDisk.replaceParam(
                userRuleItem.key,
                {
                  innerHTMLAccessCodeAfterMaxRange:
                    userRegExpOther["innerHTMLAccessCodeAfterMaxRange"],
                }
              );
            }
            return userRegexp;
          }
          if (Array.isArray(userRegExp)) {
            userRegExp = userRegExp.map((userRegexpItem) =>
              handleRegExp(userRegexpItem)
            );
          } else {
            handleRegExp(userRegExp);
          }
          if (this.$data.settingViewDetails.has(userRuleItem.key)) {
            this.$data.settingViewDetails.delete(userRuleItem.key);
          }
          this.$data.settingViewDetails.set(userRuleItem.key, viewDetails);
        }
        if (typeof userRuleItem["icon"] === "string") {
          if (userRuleItem["icon"] in NetDiskUI.src.icon) {
            /* 判断icon的值是否是图标字典中的键,是的话让该规则配置它的图标 */
            NetDiskUI.src.icon[userRuleItem["key"]] =
              NetDiskUI.src.icon[userRuleItem["icon"]];
          } else {
            NetDiskUI.src.icon[userRuleItem["key"]] = userRuleItem["icon"];
          }
        }
        if (Array.isArray(userRegExp)) {
          userRegExp = userRegExp.map((userRegexpItem) => {
            return parseRegExp(userRegexpItem);
          });
        } else {
          parseRegExp(userRegExp);
        }
        if (userRule[userRuleItem.key]) {
          /* 已存在相同key规则,追加新的 */
          userRule[userRuleItem.key].push(userRegExp);
        } else {
          if (Array.isArray(userRegExp)) {
            userRule[userRuleItem.key] = userRegExp;
          } else {
            userRule[userRuleItem.key] = [userRegExp];
          }
        }
        const AsyncFunction = Object.getPrototypeOf(
          async function () {}
        ).constructor;
        if (typeof userRuleItem.checkLinkValidityFunction === "string") {
          try {
            if (NetDiskCheckLinkValidity.netDisk[userRuleItem.key] == null) {
              NetDiskCheckLinkValidity.netDisk[userRuleItem.key] = {};
            }
            NetDiskCheckLinkValidity.netDisk[userRuleItem.key].init =
              new AsyncFunction(
                "netDiskIndex",
                "shareCode",
                "accessCode",
                userRuleItem.checkLinkValidityFunction
              ).bind(this.getBindContext(userRuleItem));
          } catch (error) {
            log.error(error);
          }
        }
        if (typeof userRuleItem.AuthorizationFunction === "string") {
          try {
            NetDiskAuthorization.netDisk[userRuleItem.key] = new AsyncFunction(
              userRuleItem.AuthorizationFunction
            ).bind(this.getBindContext(userRuleItem));
          } catch (error) {
            log.error(error);
          }
        }
        if (typeof userRuleItem.AutoFillAccessCodeFunction === "string") {
          try {
            NetDiskAutoFillAccessCode.netDisk[userRuleItem.key] =
              new AsyncFunction(
                "netDiskInfo",
                userRuleItem.AutoFillAccessCodeFunction
              ).bind(this.getBindContext(userRuleItem));
          } catch (error) {
            log.error(error);
          }
        }
        if (typeof userRuleItem.parseFunction === "string") {
          try {
            NetDiskParse.netDisk[userRuleItem.key] = new Function(
              userRuleItem.parseFunction
            ).bind(this.getBindContext(userRuleItem));
          } catch (error) {
            log.error(error);
          }
        }
      }
      return userRule;
    },
    /**
     * 获取设置菜单配置
     */
    getSettingViewDetails() {
      let viewDetailsList = [];
      this.$data.settingViewDetails.forEach((value) => {
        viewDetailsList.push(value);
      });
      return viewDetailsList;
    },
    /**
     * 初始化默认值
     */
    initDefaultValue(key, value) {
      if (GM_getValue(key) == null) {
        GM_setValue(key, value);
      }
    },
    /**
     * 获取模板规则
     * @returns
     */
    getTemplateRule() {
      /**
       * @type {NetDiskUserCustomRule}
       */
      let templateRule = {
        key: "规则名",
        icon: "图标链接或base64图片",
        regexp: [
          {
            link_innerText: "",
            link_innerHTML: "",
            shareCode: "",
            shareCodeNeedRemoveStr: "",
            uiLinkShow: "",
            blank: "",
            copyUrl: "",
          },
        ],
        setting: {
          name: "名字",
          enable: true,
          isBlank: true,
          openBlankWithCopyAccessCode: true,
        },
      };
      return this.getFormatRule(templateRule);
    },
    /**
     * 添加规则
     * @param {NetDiskUserCustomRule} userRule
     */
    addRule(userRule) {
      let localRule = this.getRule();
      localRule.push(userRule);
      GM_setValue(NetDiskCustomRules.key, localRule);
    },
    /**
     * 设置规则到本地
     * @param {string} oldRuleKey 旧规则的键名
     * @param {NetDiskUserCustomRule[]|NetDiskUserCustomRule} userRule
     */
    setRule(oldRuleKey, userRule) {
      if (Array.isArray(userRule)) {
        GM_setValue(NetDiskCustomRules.key, userRule);
      } else {
        let localRule = this.getRule();
        let findRuleIndex = localRule.findIndex(
          (item) => item.key === oldRuleKey
        );
        if (findRuleIndex !== -1) {
          localRule[findRuleIndex] = null;
          localRule[findRuleIndex] = userRule;
        } else {
          log.error(["覆盖规则失败", userRule]);
          Qmsg.error("覆盖规则失败");
          return false;
        }
        this.setRule(oldRuleKey, localRule);
      }
    },
    /**
     * 删除单条规则
     * @param {string} ruleKey 规则的key名
     */
    deleteRule(ruleKey) {
      let localRule = this.getRule();
      let findIndex = localRule.findIndex((rule) => rule.key === ruleKey);
      if (findIndex !== -1) {
        localRule.splice(findIndex, 1);
        this.setRule(ruleKey, localRule);
        return true;
      } else {
        return false;
      }
    },
    /**
     * 清空规则
     */
    clearRule() {
      GM_deleteValue(NetDiskCustomRules.key);
    },
    /**
     * 获取规则
     * @param {?string} key
     * @returns {(NetDiskUserCustomRule|NetDiskUserCustomRule)[]}
     */
    getRule(key) {
      if (typeof key === "string") {
        return GM_getValue(NetDiskCustomRules.key, []).find(
          (item) => item.key === key
        );
      }
      return GM_getValue(NetDiskCustomRules.key, []);
    },
    /**
     * 获取格式化后的规则
     * @param {?string} rule
     */
    getFormatRule(rule) {
      return JSON.stringify(rule || this.getRule(), void 0, 4);
    },
  };

  /** 弹窗UI界面 */
  const NetDiskUI = {
    /** 元素对象实例 */
    Alias: {
      /**
       * 链接弹窗的对象
       * @type {PopsCallResult}
       */
      uiLinkAlias: void 0,
      /**
       * 历史匹配记录弹窗的对象
       * @type {PopsCallResult}
       */
      historyAlias: void 0,
      /**
       * 设置弹窗的对象
       */
      settingAlias: void 0,
    },
    /**
     * 已匹配到的网盘图标字典
     * @type {Set<string>}
     */
    isMatchedNetDiskIconMap: new Set(),
    /**
     * 高度和宽度大小
     */
    size: parseInt(GM_getValue("size", 50)),
    /**
     * 按钮透明度
     */
    opacity: parseFloat(GM_getValue("opacity", 1)),
    /**
     * 是否正在文本匹配中
     */
    isHandleMatch: false,
    /**
     * 是否正在循环切换按钮背景
     */
    isSwitchRandomBackground: false,
    /**
     * 是否默认弹窗可以拖拽
     */
    defaultPCDrag: false,
    /**
     * 是否默认弹窗拖拽距离
     */
    defaultPCDragLimit: true,
    /**
     * 是否默认开启亚克力效果
     */
    defaultPopsAcrylic: false,
    /**
     * 点击弹窗遮罩层是否可以关闭弹窗
     */
    defaultClickMaskToCloseDialog: false,
    /**
     * 是否默认禁用弹窗弹出后背景可以滚动
     */
    defaultForbiddenScroll: false,
    /**
     * 默认弹窗动画
     */
    defaultAnimation: "pops-anim-fadein-zoom",
    /**
     * folder默认排序的属性名
     */
    defaultSortName: "fileName",
    /**
     * folder是否降序排序
     */
    defaultSortDesc: false,
    /**
     * 弹窗的配置
     * 规定格式
     * {
     *  PC:{
     *    width: "",
     *    height: "",
     *  },
     *  Mobile: {
     *    width: "",
     *    height: "",
     *  }
     * }
     */
    popsStyle: {
      /**
       * 天翼云需要登录的提示
       */
      tianYiYunLoginTip: {
        PC: {
          width: "30vw",
          height: "280px",
        },
        Mobile: {
          width: "80vw",
          height: "250px",
        },
      },
      /**
       * 坚果云需要登录的提示
       */
      jianGuoYunLoginTip: {
        PC: {
          width: "350px",
          height: "200px",
        },
        Mobile: {
          width: "350px",
          height: "200px",
        },
      },
      /**
       * 设置
       */
      settingView: {
        PC: {
          width: "800px",
          height: "600px",
        },
        Mobile: {
          width: "92vw",
          height: "80vh",
        },
      },
      /**
       * 设置默认值的界面
       */
      setDefaultValueView: {
        PC: {
          width: "350px",
          height: "200px",
        },
        Mobile: {
          width: "350px",
          height: "200px",
        },
      },
      /**
       * (主)网盘链接界面
       */
      mainView: {
        PC: {
          width: "500px",
          height: "100%",
        },
        Mobile: {
          width: "88dvw",
          height: "50dvh",
        },
      },
      /**
       * (主)网盘链接界面-小窗
       */
      mainViewSmallWindow: {
        PC: {
          width:
            GM_getValue(
              "netdisk-ui-small-window-width",
              NetDiskData.defaultSmallWindowWidth
            ) + "px",
          height: "auto",
        },
        Mobile: {
          width:
            GM_getValue(
              "netdisk-ui-small-window-width",
              NetDiskData.defaultSmallWindowWidth
            ) + "px",
          height: "auto",
        },
      },
      /**
       * 单文件直链弹窗
       */
      oneFileStaticView: {
        PC: {
          width: "550px",
          height: "350px",
        },
        Mobile: {
          width: "88dvw",
          height: "300px",
        },
      },
      /**
       * 多文件直链弹窗
       */
      moreFileStaticView: {
        PC: {
          width: "700px",
          height: "600px",
        },
        Mobile: {
          width: "88vw",
          height: "500px",
        },
      },
      /**
       * 新密码、错误密码输入弹窗
       */
      inputNewAccessCodeView: {
        PC: {
          width: "400px",
          height: "200px",
        },
        Mobile: {
          width: "88vw",
          height: "160px",
        },
      },
      /**
       * 历史存储记录弹窗
       */
      historyMatchView: {
        PC: {
          width: "50vw",
          height: "65vh",
        },
        Mobile: {
          width: "88vw",
          height: "60vh",
        },
      },
      /**
       * 自定义规则的弹窗
       */
      customRulesView: {
        PC: {
          width: "50vw",
          height: "65vh",
        },
        Mobile: {
          width: "88vw",
          height: "60vh",
        },
      },
      /**
       * 自定义规则的调试视图
       */
      customRuleDebugView: {
        PC: {
          width: "55dvw",
          height: "70dvh",
        },
        Mobile: {
          width: "88vw",
          height: "70vh",
        },
      },
      /**
       * 主动识别的弹窗
       */
      matchPasteTextView: {
        PC: {
          width: "50vw",
          height: "65vh",
        },
        Mobile: {
          width: "88vw",
          height: "60vh",
        },
      },
      /**
       * 访问码规则弹窗
       */
      accessCodeRuleView: {
        PC: {
          width: "50vw",
          height: "65vh",
        },
        Mobile: {
          width: "88vw",
          height: "60vh",
        },
      },
      /**
       * 访问码规则添加/修改/删除
       */
      accessCodeRuleEditView: {
        PC: {
          width: "44vw",
          height: "50vh",
        },
        Mobile: {
          width: "70vw",
          height: "45vh",
        },
      },
    },
    src: {
      /**
       * 图标RESOURCE_ICON
       * 从图标库中引入并覆盖
       */
      icon: {},
    },
    /**
     * 初始化Qmsg的配置
     */
    initQmsg() {
      Qmsg.config({
        position: GM_getValue("qmsg-position", "top"),
        html: true,
        maxNums: parseInt(GM_getValue("qmsg-maxnums", 3)),
        autoClose: true,
        showClose: false,
        showReverse: GM_getValue("qmsg-showreverse", false),
      });
    },
    /**
     * 悬浮按钮  双击打开主界面,长按打开设置(不能移动,移动就不打开,只是移动按钮
     */
    suspension: {
      /** @type {HTMLDivElement} */
      suspensionNode: null,
      /** 是否已显示 */
      isShow: false,
      /** 是否已设置事件 */
      isSetEvent: false,
      /** 是否正在切换背景 */
      isRandBg: false,
      /**
       * 显示悬浮按钮
       */
      show() {
        if (!this.isShow) {
          this.isShow = true;
          this.createUI();
          this.setSuspensionPosition();
        }
        if (!this.isSetEvent) {
          this.isSetEvent = true;
          this.setSuspensionEvent();
          this.setResizeEventListener();
        }
        this.backgroundSwitch();
        this.showSuspension();
      },
      showSuspension() {
        this.suspensionNode.style.display = "";
      },
      hideSuspension() {
        this.suspensionNode.style.display = "none";
      },
      /**
       * 判断当前是否是顶部窗口
       * @returns {boolean}
       */
      isTopWindow() {
        return unsafeWindow.self.window === unsafeWindow.top.window;
      },
      /**
       * 创建UI界面
       */
      createUI() {
        if (NetDiskUI.size < 15) {
          /* 大小不能小于 15px */
          GM_setValue("size", 15);
          NetDiskUI.size = 15;
        }
        if (NetDiskUI.size > 250) {
          /* 大小不能大于 250px */
          GM_setValue("size", 250);
          NetDiskUI.size = 250;
        }
        if (NetDiskUI.opacity < 0.1) {
          /* 透明度不能小于 0.1 */
          GM_setValue("opacity", 0.1);
          NetDiskUI.opacity = 0.1;
        }
        if (NetDiskUI.opacity > 1.0) {
          /* 透明度不能大于 1.0 */
          GM_setValue("opacity", 1);
          NetDiskUI.opacity = 1;
        }
        let $shadowContainer = DOMUtils.createElement("div", {
          className: "whitesev-suspension-shadow-container",
        });
        let $shadowRoot = $shadowContainer.attachShadow({ mode: "open" });
        this.suspensionNode = DOMUtils.createElement(
          "div",
          {
            id: "whitesevSuspensionId",
            className: "whitesevSuspension",
            innerHTML: `
            <style type="text/css">

            ${this.getCSS()}

            </style>
            <div class="whitesevSuspensionMain">
              <div class="whitesevSuspensionFloor">
                <div class="netdisk"></div>
              </div>
            </div>
            `,
          },
          {
            style: `
            width: ${NetDiskUI.size}px;
            height: ${NetDiskUI.size}px;
            opacity: ${NetDiskUI.opacity}
          `,
          }
        );
        $shadowRoot.appendChild(this.suspensionNode);
        document.body.appendChild($shadowContainer);
      },
      /**
       * 显示设置
       */
      showSettingView() {
        if (NetDiskUI.Alias.settingAlias) {
          log.error("设置界面已存在");
          Qmsg.error("设置界面已存在");
          return;
        }
        const that = this;
        /**
         *
         * @param {string} text
         * @param {string} key
         * @param {boolean} defaultValue
         * @param {(event:Event,enable:boolean)=>void} callback
         * @param {string} description
         * @returns
         */
        function getSwtichDetail(
          text,
          key,
          defaultValue,
          callback,
          description
        ) {
          defaultValue = Boolean(defaultValue);
          return {
            text: text,
            type: "switch",
            description: description,
            attributes: {
              "data-key": key,
              "data-default-value": defaultValue,
            },
            getValue() {
              return Boolean(GM_getValue(key, defaultValue));
            },
            callback(event, value) {
              GM_setValue(key, Boolean(value));
              if (typeof callback === "function") {
                callback(event, value);
              }
            },
          };
        }
        /**
         *
         * @param {string} text
         * @param {string} description
         * @param {string} key
         * @param {number} defaultValue
         * @param {number} min
         * @param {number} max
         * @param {?number} step
         * @param {?(event: PointerEvent,value: number)=> void} callback
         * @param {?(value: number)=> string|number} getToolTipContent
         * @returns
         */
        function getSliderDetail(
          text,
          description,
          key,
          defaultValue,
          min,
          max,
          step,
          callback,
          getToolTipContent
        ) {
          return {
            text: text,
            type: "slider",
            description: description,
            attributes: {
              "data-key": key,
              "data-default-value": defaultValue,
            },
            getValue() {
              return GM_getValue(key, defaultValue);
            },
            callback(event, value) {
              GM_setValue(key, value);
              if (typeof callback === "function") {
                callback(event, value);
              }
            },
            getToolTipContent(value) {
              if (typeof getToolTipContent === "function") {
                return getToolTipContent(value);
              } else {
                return value;
              }
            },
            min: min,
            max: max,
            step: step,
          };
        }
        /**
         *
         * @param {string} text
         * @param {string} description
         * @param {string} key
         * @param {any} defaultValue
         * @param {{
         * text:string,value:string
         * }[]} data
         * @param {(event:PointerEvent, isSelectedValue: any, isSelectedText:string)=>void} callback
         */
        function getSelectDetail(
          text,
          description,
          key,
          defaultValue,
          data,
          callback
        ) {
          return {
            text: text,
            type: "select",
            description: description,
            attributes: {
              "data-key": key,
              "data-default-value": defaultValue,
            },
            getValue() {
              return GM_getValue(key, defaultValue);
            },
            callback(event, isSelectedValue, isSelectedText) {
              GM_setValue(key, isSelectedValue);
              if (typeof callback === "function") {
                callback(event, isSelectedValue, isSelectedText);
              }
            },
            data: data,
          };
        }
        /**
         * @type {PopsPanelContentConfig[]}
         */
        let contentDetails = [
          {
            id: "netdisk-panel-config-all-setting",
            title: "总设置",
            isDefault: true,
            forms: [
              {
                className: "netdisk-panel-forms-pops",
                text: "弹窗",
                type: "forms",
                forms: [
                  getSelectDetail(
                    "动画",
                    `显示/关闭的动画效果,默认: ${NetDiskUI.defaultAnimation}`,
                    "popsAnimation",
                    NetDiskUI.defaultAnimation,
                    [
                      {
                        value: "",
                        text: "无",
                      },
                      {
                        value: "pops-anim-spread",
                        text: "spread",
                      },
                      {
                        value: "pops-anim-shake",
                        text: "shake",
                      },
                      {
                        value: "pops-anim-rolling-left",
                        text: "rolling-left",
                      },
                      {
                        value: "pops-anim-rolling-right",
                        text: "rolling-right",
                      },
                      {
                        value: "pops-anim-slide-top",
                        text: "slide-top",
                      },
                      {
                        value: "pops-anim-slide-bottom",
                        text: "slide-bottom",
                      },
                      {
                        value: "pops-anim-slide-left",
                        text: "slide-left",
                      },
                      {
                        value: "pops-anim-slide-right",
                        text: "slide-right",
                      },
                      {
                        value: "pops-anim-fadein",
                        text: "fadein",
                      },
                      {
                        value: "pops-anim-fadein-zoom",
                        text: "fadein-zoom",
                      },
                      {
                        value: "pops-anim-fadein-alert",
                        text: "fadein-alert",
                      },
                      {
                        value: "pops-anim-don",
                        text: "don",
                      },
                      {
                        value: "pops-anim-roll",
                        text: "roll",
                      },
                      {
                        value: "pops-anim-sandra",
                        text: "sandra",
                      },
                      {
                        value: "pops-anim-gather",
                        text: "gather",
                      },
                    ]
                  ),
                  getSwtichDetail(
                    "点击弹窗遮罩层关闭弹窗",
                    "clickMaskToCloseDialog",
                    NetDiskUI.defaultClickMaskToCloseDialog,
                    void 0,
                    "点击遮罩层触发关闭弹窗事件"
                  ),
                  getSwtichDetail(
                    "窗口拖拽",
                    "pcDrag",
                    NetDiskUI.defaultPCDrag,
                    void 0,
                    "长按标题栏可拖拽移动弹窗"
                  ),
                  getSwtichDetail(
                    "限制拖拽距离",
                    "pcDragLimit",
                    NetDiskUI.defaultPCDragLimit,
                    void 0,
                    "只能在浏览器的可视窗口内拖动"
                  ),
                  getSwtichDetail(
                    "亚克力效果",
                    "popsAcrylic",
                    NetDiskUI.defaultPopsAcrylic,
                    void 0,
                    ""
                  ),
                ],
              },
              {
                className: "netdisk-panel-forms-pops-folder",
                text: "文件弹窗",
                type: "forms",
                forms: [
                  getSelectDetail(
                    "排序名",
                    "当前的规则",
                    "pops-folder-sort-name",
                    NetDiskUI.defaultSortName,
                    [
                      {
                        value: "fileName",
                        text: "文件名",
                      },
                      {
                        value: "latestTime",
                        text: "修改时间",
                      },
                      {
                        value: "fileSize",
                        text: "大小",
                      },
                    ]
                  ),
                  getSelectDetail(
                    "排序规则",
                    "当前的规则",
                    "pops-folder-sort-is-desc",
                    NetDiskUI.defaultSortDesc,
                    [
                      {
                        value: false,
                        text: "升序",
                      },
                      {
                        value: true,
                        text: "降序",
                      },
                    ]
                  ),
                ],
              },
              {
                text: "小图标导航",
                type: "forms",
                forms: [
                  getSwtichDetail(
                    "点击定位分享码",
                    "pops-netdisk-icon-click-event-find-sharecode",
                    true,
                    NetDiskData.iconDefaultClickEventToFindShareCode,
                    "自动滚动页面至包含分享码的元素"
                  ),
                  getSwtichDetail(
                    "选中分享码",
                    "pops-netdisk-icon-click-event-find-sharecode-with-select",
                    true,
                    NetDiskData.iconDefaultClickEventToFindShareCodeWithSelect,
                    "使用光标选中分享码/元素"
                  ),
                  getSwtichDetail(
                    "循环定位",
                    "pops-netdisk-icon-click-event-loop-find-sharecode",
                    true,
                    NetDiskData.iconDefaultClickEventToFindShareCodeByLoop,
                    "关闭则是每一个元素只定位一次"
                  ),
                ],
              },
              {
                text: "悬浮按钮",
                type: "forms",
                forms: [
                  getSliderDetail(
                    "大小",
                    "悬浮按钮的大小,默认: 50",
                    "size",
                    50,
                    15,
                    250,
                    void 0,
                    (event, value) => {
                      NetDiskUI.size = parseInt(value);
                      if (NetDiskUI.suspension.isShow) {
                        DOMUtils.css(NetDiskUI.suspension.suspensionNode, {
                          width: NetDiskUI.size,
                          height: NetDiskUI.size,
                        });
                        NetDiskUI.suspension.setSuspensionPosition();
                      }
                    },
                    (value) => {
                      return `${value}px`;
                    }
                  ),
                  getSliderDetail(
                    "透明度",
                    "值越小越透明,默认: 1",
                    "opacity",
                    1,
                    0.1,
                    1,
                    0.1,
                    (event, value) => {
                      NetDiskUI.opacity = parseFloat(value);
                      if (NetDiskUI.suspension.isShow) {
                        DOMUtils.css(NetDiskUI.suspension.suspensionNode, {
                          opacity: NetDiskUI.opacity,
                        });
                      }
                    }
                  ),
                  getSliderDetail(
                    "背景轮播时间",
                    "淡入/淡出的时间,默认: 1500",
                    "randbg-time",
                    1500,
                    0,
                    10000,
                    100,
                    void 0,
                    (value) => {
                      return `${value}ms`;
                    }
                  ),
                  getSliderDetail(
                    "背景显示时间",
                    "图标显示的持续时间,默认: 1200",
                    "randbg-show-time",
                    1200,
                    0,
                    10000,
                    100,
                    void 0,
                    (value) => {
                      return `${value}ms`;
                    }
                  ),
                  getSwtichDetail(
                    "吸附边缘",
                    "suspended-button-adsorption-edge",
                    false,
                    void 0,
                    "移动悬浮按钮松开后自动吸附边缘"
                  ),
                ],
              },
              {
                className: "netdisk-panel-forms-toast",
                text: "Toast",
                type: "forms",
                forms: [
                  getSelectDetail(
                    "位置",
                    "Toast显示在九宫格的位置,默认: 中间",
                    "qmsg-position",
                    "top",
                    [
                      {
                        value: "topleft",
                        text: "左上角",
                      },
                      {
                        value: "top",
                        text: "顶部",
                      },
                      {
                        value: "topright",
                        text: "右上角",
                      },
                      {
                        value: "left",
                        text: "左边",
                      },
                      {
                        value: "center",
                        text: "中间",
                      },
                      {
                        value: "right",
                        text: "右边",
                      },
                      {
                        value: "bottomleft",
                        text: "左下角",
                      },
                      {
                        value: "bottom",
                        text: "底部",
                      },
                      {
                        value: "bottomright",
                        text: "右下角",
                      },
                    ],
                    NetDiskUI.initQmsg
                  ),
                  getSelectDetail(
                    "同时最多显示的数量",
                    "默认: 3",
                    "qmsg-maxnums",
                    3,
                    [
                      {
                        value: 1,
                        text: "1",
                      },
                      {
                        value: 2,
                        text: "2",
                      },
                      {
                        value: 3,
                        text: "3",
                      },
                      {
                        value: 4,
                        text: "4",
                      },
                      {
                        value: 5,
                        text: "5",
                      },
                    ],
                    NetDiskUI.initQmsg
                  ),
                  getSwtichDetail(
                    "逆序弹出",
                    "qmsg-showreverse",
                    false,
                    NetDiskUI.initQmsg,
                    "默认是自上往下显示Toast,逆序则是自下往上显示Toast"
                  ),
                ],
              },
              {
                className: "netdisk-panel-forms-small-window",
                text: "小窗模式",
                type: "forms",
                forms: [
                  getSliderDetail(
                    "宽度",
                    "设置小窗宽度(px),默认: 250",
                    "netdisk-ui-small-window-width",
                    NetDiskData.defaultSmallWindowWidth,
                    NetDiskData.smallWindowMinWidth,
                    NetDiskData.smallWindowMaxWidth,
                    1,
                    void 0,
                    (value) => {
                      return `${value}px`;
                    }
                  ),
                  getSliderDetail(
                    "高度",
                    "设置小窗最大高度(px),默认: 200",
                    "netdisk-ui-small-window-max-height",
                    NetDiskData.defaultSmallWindowMaxHeight,
                    NetDiskData.smallWindowMinMaxHeight,
                    NetDiskData.smallWindowMaxMaxHeight,
                    1,
                    void 0,
                    (value) => {
                      return `${value}px`;
                    }
                  ),
                ],
              },
              {
                className: "netdisk-panel-history-match",
                text: "匹配记录",
                type: "forms",
                forms: [
                  getSelectDetail(
                    "排序规则",
                    "",
                    "netdisk-history-match-ordering-rule",
                    "按 更新时间 - 降序",
                    [
                      {
                        value: "按 记录时间 - 升序",
                        text: "按 记录时间 - 升序",
                      },
                      {
                        value: "按 记录时间 - 降序",
                        text: "按 记录时间 - 降序",
                      },
                      {
                        value: "按 更新时间 - 升序",
                        text: "按 更新时间 - 升序",
                      },
                      {
                        value: "按 更新时间 - 降序",
                        text: "按 更新时间 - 降序",
                      },
                    ]
                  ),
                  {
                    text: "修复存储记录",
                    type: "button",
                    description:
                      "如果【匹配记录】弹窗打不开,可能是存储的数据缺失某些字段,可尝试点击此处进行修复",
                    buttonIconIsLoading: false,
                    buttonType: "primary",
                    buttonText() {
                      return "修复";
                    },
                    callback(event) {
                      try {
                        const { count, repairCount } =
                          NetDiskUI.netDiskHistoryMatch.checkAndRepairLocalData();
                        if (repairCount === 0) {
                          Qmsg.info(`不存在需要修复的数据`);
                        } else {
                          Qmsg.success(
                            `共计: ${count} 条,修复${repairCount}条`
                          );
                        }
                      } catch (error) {
                        Qmsg.error("修复异常:" + error.toString());
                      }
                    },
                  },
                  getSwtichDetail(
                    "保存匹配记录",
                    "saveMatchNetDisk",
                    false,
                    void 0,
                    "将匹配到的链接信息进行本地存储,可点击【油猴菜单-⚙ 历史匹配记录】进行查看"
                  ),
                ],
              },
              {
                className: "netdisk-panel-forms-function",
                text: "功能",
                type: "forms",
                forms: [
                  getSelectDetail(
                    "行为模式",
                    "匹配到链接时触发的UI执行",
                    "netdisk-behavior-mode",
                    NetDiskData.defaultNetdiskBehaviorMode,
                    [
                      {
                        text: "悬浮按钮+小窗",
                        value: "suspension_smallwindow",
                      },
                      {
                        text: "悬浮按钮+大窗",
                        value: "suspension_window",
                      },
                      {
                        text: "小窗",
                        value: "smallwindow",
                      },
                    ],
                    void 0
                  ),
                  getSliderDetail(
                    "匹配间隔",
                    "匹配文本完毕后的延迟xxx秒允许下一次匹配",
                    "delaytime",
                    NetDiskData.defaultMatchIntervalTime,
                    NetDiskData.matchIntervalMinTime,
                    NetDiskData.matchIntervalMaxTime,
                    0.1,
                    void 0,
                    (value) => {
                      return `${value}s`;
                    }
                  ),
                  getSelectDetail(
                    "匹配类型",
                    "匹配的文本类型",
                    "pageMatchRange",
                    "all",
                    [
                      {
                        value: "all",
                        text: "全部",
                      },
                      {
                        value: "innerText",
                        text: "普通文本",
                      },
                      {
                        value: "innerHTML",
                        text: "超文本",
                      },
                    ]
                  ),
                  getSwtichDetail(
                    "读取剪贴板",
                    "readClipboard",
                    false,
                    void 0,
                    "Api兼容性查看:<a href='https://caniuse.com/mdn-api_permissions_permission_clipboard-read' target='_blank'>读取剪贴板权限申请</a>、<a href='https://caniuse.com/mdn-api_clipboard_readtext' target='_blank'>直接读取剪贴板</a>"
                  ),
                  getSwtichDetail(
                    "自动输入访问码",
                    "autoFillAccessCode",
                    false,
                    void 0,
                    "通过主动点击链接跳转时,会自动输入网盘访问码"
                  ),
                  getSwtichDetail(
                    "获取重定向后的直链",
                    "getTheDirectLinkAfterRedirection",
                    false,
                    void 0,
                    "对获取的链接再进行一次重定向获取链接"
                  ),
                  getSwtichDetail(
                    "允许匹配当前URL",
                    "allowMatchLocationHref",
                    false,
                    void 0,
                    "匹配window.location.href"
                  ),
                  getSwtichDetail(
                    "添加元素时进行匹配",
                    "isAddedNodesToMatch",
                    false,
                    void 0,
                    "当监听到页面添加元素时才进行匹配文本"
                  ),
                ],
              },
              {
                className: "netdisk-panel-forms-share-code",
                text: "分享码相关",
                type: "forms",
                forms: [
                  getSliderDetail(
                    "相同系数",
                    "例如分享码: aaaaaaaabb,它的相同系数是0.8,设置相同系数≥0.8时会被排除",
                    "excludeIdenticalSharedCodesCoefficient",
                    NetDiskData.defaultExcludeIdenticalSharedCodesCoefficient,
                    NetDiskData.excludeIdenticalSharedCodesCoefficientMin,
                    NetDiskData.excludeIdenticalSharedCodesCoefficientMax,
                    0.01,
                    void 0,
                    (value) => {
                      return value;
                    }
                  ),
                  getSwtichDetail(
                    "排除分享码",
                    "excludeIdenticalSharedCodes",
                    false,
                    void 0,
                    "启用后会根据【相同系数】排除掉匹配到的分享码"
                  ),
                ],
              },
            ],
          },
        ];

        let shortcutDetails = {
          className: "netdisk-panel-forms-shortcut-keys",
          text: "快捷键",
          type: "forms",
          forms: [],
        };
        let shortcurFormsDetailsList = [
          {
            text: "【打开】⚙ 设置",
            key: "netdisk-keyboard-open-setting",
          },
          {
            text: "【打开】⚙ 历史匹配记录",
            key: "netdisk-keyboard-open-history-matching-records",
          },
          {
            text: "【打开】⚙ 访问码规则",
            key: "netdisk-keyboard-open-accesscode-rule",
          },
          {
            text: "【打开】⚙ 用户自定义规则",
            key: "netdisk-keyboard-open-user-rule",
          },
          {
            text: "【打开】⚙ 主动识别文本",
            key: "netdisk-keyboard-open-proactively-recognize-text",
          },
        ];
        shortcurFormsDetailsList.forEach((item) => {
          shortcutDetails.forms.push({
            text: item.text,
            type: "button",
            attributes: {
              "data-key": item.key,
              "data-default-value": "暂无快捷键",
            },
            buttonIcon: "keyboard",
            buttonIconIsLoading: false,
            buttonType: "default",
            buttonText() {
              return NetDiskShortcut.getShowText(
                this.attributes["data-key"],
                this.attributes["data-default-value"]
              );
            },
            callback(event) {
              NetDiskShortcut.buttonClickCallBack(
                event,
                this.attributes["data-key"],
                this.attributes["data-default-value"]
              );
            },
          });
        });
        contentDetails[0].forms.push(shortcutDetails);
        /**
         * 追加网盘设置
         */
        function addNetDiskSetting() {
          /**
           * @type {NetDiskSettingMenuDetails[]}
           */
          let netDiskDetailsList = [
            {
              type: "百度网盘",
              key: "baidu",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              checkLinkValidity: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
              ownFormList: [
                {
                  text: "第三方解析站",
                  type: "forms",
                  forms: [
                    {
                      text: "启用解析站",
                      type: "switch",
                      description:
                        "开源项目:<a href='https://github.com/yuantuo666/baiduwp-php' target='_blank'>https://github.com/yuantuo666/baiduwp-php</a>",
                      attributes: {
                        "data-key": "baidu-static-enable",
                        "data-default-value": false,
                      },
                      getValue() {
                        return Boolean(
                          GM_getValue(
                            this.attributes["data-key"],
                            this.attributes["data-default-value"]
                          )
                        );
                      },
                      callback(event, value) {
                        GM_setValue(
                          this.attributes["data-key"],
                          Boolean(value)
                        );
                      },
                    },
                    {
                      text: "跳转时复制链接",
                      type: "switch",
                      description: "跳转至解析站时复制百度网盘链接",
                      attributes: {
                        "data-key": "baidu-baiduwp-php-copy-url",
                        "data-default-value": false,
                      },
                      getValue() {
                        return Boolean(
                          GM_getValue(
                            this.attributes["data-key"],
                            this.attributes["data-default-value"]
                          )
                        );
                      },
                      callback(event, value) {
                        GM_setValue(
                          this.attributes["data-key"],
                          Boolean(value)
                        );
                      },
                    },
                    {
                      text: "网址",
                      type: "input",
                      description: "解析站的网址Url",
                      attributes: {
                        "data-key": "baidu-baiduwp-php-url",
                        "data-default-value": "",
                      },
                      getValue() {
                        return GM_getValue(
                          this.attributes["data-key"],
                          this.attributes["data-default-value"]
                        );
                      },
                      callback(event, value) {
                        GM_setValue(this.attributes["data-key"], value);
                      },
                      placeholder:
                        "使用了baiduwp-php源码的网站,例如:https://www.example.com/",
                    },
                    {
                      text: "表单参数",
                      type: "input",
                      description: "跳转时携带的参数",
                      attributes: {
                        "data-key": "baidu-baiduwp-php-post-form",
                        "data-default-value": "",
                      },
                      getValue() {
                        return GM_getValue(
                          this.attributes["data-key"],
                          this.attributes["data-default-value"]
                        );
                      },
                      callback(event, value) {
                        GM_setValue(this.attributes["data-key"], value);
                      },
                      placeholder:
                        "POST表单,例如:surl={#shareCode#}&pwd={#accessCode#}&password=",
                    },
                  ],
                },
              ],
            },
            {
              type: "蓝奏云",
              key: "lanzou",
              enable: true,
              parseMoreFile: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              checkLinkValidity: true,
              isForward: true,
              isForwardBlankLink: true,
              isForwardDownloadLink: true,
              schemeUri: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
              ownFormList: [
                {
                  text: "其它配置",
                  type: "forms",
                  forms: [
                    {
                      text: "蓝奏云域名",
                      type: "input",
                      attributes: {
                        "data-key": "lanzou-host-name",
                        "data-default-value":
                          NetDiskData.lanzou_defaultHostName,
                      },
                      getValue() {
                        return GM_getValue(
                          this.attributes["data-key"],
                          this.attributes["data-default-value"]
                        );
                      },
                      callback(event, value) {
                        GM_setValue(this.attributes["data-key"], value);
                      },
                      placeholder: `例如:${NetDiskData.lanzou_defaultHostName}`,
                    },
                  ],
                },
              ],
            },
            {
              type: "蓝奏云优享",
              key: "lanzouyx",
              enable: true,
              parseMoreFile: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              checkLinkValidity: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "天翼云",
              key: "tianyiyun",
              enable: true,
              parseMoreFile: true,
              parseFileDescription: "需要登录才可以下载",
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "中国移动云盘",
              key: "hecaiyun",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "阿里云",
              key: "aliyun",
              enable: true,
              parseMoreFile: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "文叔叔",
              key: "wenshushu",
              enable: true,
              parseOneFile: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "奶牛",
              key: "nainiu",
              enable: true,
              parseMoreFile: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "123盘",
              key: "_123pan",
              enable: true,
              parseMoreFile: true,
              parseFileDescription: "需要登录才可下载>100MB的文件",
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "微云",
              key: "weiyun",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "迅雷云盘",
              key: "xunlei",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "115网盘",
              key: "_115pan",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "城通网盘",
              key: "chengtong",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "夸克网盘",
              key: "kuake",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "BT磁力",
              key: "magnet",
              enable: true,
              isBlank: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
            },
            {
              type: "坚果云",
              key: "jianguoyun",
              enable: true,
              parseMoreFile: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "OneDrive",
              key: "onedrive",
              enable: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
            {
              type: "UC网盘",
              key: "uc",
              enable: true,
              parseMoreFile: true,
              isBlank: true,
              openBlankWithCopyAccessCode: true,
              isForward: true,
              isForwardDownloadLink: true,
              isForwardBlankLink: true,
              schemeUri: true,
              checkLinkValidity: true,
              innerTextAccessCodeBeforeMaxRange: 20,
              innerTextAccessCodeAfterMaxRange: 10,
              innerHTMLAccessCodeBeforeMaxRange: 100,
              innerHTMLAccessCodeAfterMaxRange: 15,
            },
          ];
          let userCustomDetailsList =
            NetDiskCustomRules.getSettingViewDetails();
          if (userCustomDetailsList.length) {
            /* 添加自定义的设置 */
            netDiskDetailsList = [
              ...netDiskDetailsList,
              ...userCustomDetailsList,
            ];
          }

          netDiskDetailsList.forEach((item) => {
            let formsList = [];
            if (
              typeof item.innerTextAccessCodeBeforeMaxRange === "number" ||
              typeof item.innerTextAccessCodeAfterMaxRange === "number"
            ) {
              let matchTextList = [];
              if (typeof item.innerTextAccessCodeBeforeMaxRange === "number") {
                matchTextList.push(
                  getSliderDetail(
                    "间隔前",
                    "提取码间隔前的字符长度",
                    "innerText_" + item.key,
                    item.innerTextAccessCodeBeforeMaxRange,
                    0,
                    100
                  )
                );
              }
              if (typeof item.innerTextAccessCodeAfterMaxRange === "number") {
                matchTextList.push(
                  getSliderDetail(
                    "间隔后",
                    "提取码间隔后的字符长度",
                    "accessCode_after_text_" + item.key,
                    item.innerTextAccessCodeAfterMaxRange,
                    0,
                    100
                  )
                );
              }
              if (matchTextList.length) {
                formsList.push({
                  text: "提取码文本匹配Text",
                  type: "forms",
                  forms: matchTextList,
                });
              }
            }
            if (
              typeof item.innerHTMLAccessCodeBeforeMaxRange === "number" ||
              typeof item.innerHTMLAccessCodeAfterMaxRange === "number"
            ) {
              let matchTextList = [];
              if (typeof item.innerHTMLAccessCodeBeforeMaxRange === "number") {
                matchTextList.push(
                  getSliderDetail(
                    "间隔前",
                    "提取码间隔前的字符长度",
                    "innerHTML_" + item.key,
                    item.innerHTMLAccessCodeBeforeMaxRange,
                    0,
                    500
                  )
                );
              }

              if (typeof item.innerHTMLAccessCodeAfterMaxRange === "number") {
                matchTextList.push(
                  getSliderDetail(
                    "间隔后",
                    "提取码间隔后的字符长度",
                    "accessCode_after_html_" + item.key,
                    item.innerHTMLAccessCodeAfterMaxRange,
                    0,
                    200
                  )
                );
              }
              if (matchTextList.length) {
                formsList.push({
                  text: "提取码文本匹配HTML",
                  type: "forms",
                  forms: matchTextList,
                });
              }
            }
            if (
              item.enable != null ||
              item.isBlank != null ||
              item.parseOneFile != null ||
              item.parseMoreFile != null
            ) {
              /**
               * @type {PopsPanelFormsTotalDetails[]}
               */
              let functionFormsList = [];
              /* 是否启用 */
              if (item.enable != null) {
                functionFormsList.push({
                  text: "启用",
                  type: "switch",
                  description: "开启可匹配该网盘",
                  attributes: {
                    "data-key": `${item.key}-enable`,
                    "data-default-value": true,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                    if (item.key != null && NetDisk.regular[item.key] != null) {
                      NetDisk.regular[item.key].forEach((__, index) => {
                        NetDisk.regular[item.key][index].enable = value;
                      });
                    }
                  },
                });
              }
              /* 新标签页打开 */
              if (typeof item.isBlank === "boolean") {
                functionFormsList.push({
                  text: "新标签页打开",
                  type: "switch",
                  description: "关闭为默认复制链接",
                  attributes: {
                    "data-key": `${item.key}-open-enable`,
                    "data-default-value": false,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                    if (item.parseOneFile || item.parseMoreFile) {
                      let $shadowRoot = event.target.getRootNode();
                      /* 新标签页打开和单/文件解析只能开启一个 */
                      /**
                       * @type {HTMLInputElement}
                       */
                      let checkboxElement = $shadowRoot.querySelector(
                        `li[data-key="${item.key}-static-enable"] input[type=checkbox]`
                      );
                      let checkboxCoreElement = $shadowRoot.querySelector(
                        `li[data-key="${item.key}-static-enable"] span.pops-panel-switch__core`
                      );
                      if (value == true && checkboxElement.checked == true) {
                        checkboxCoreElement.click();
                      }
                    }
                  },
                });
              }
              /* 新标签页打开时复制访问码 */
              if (typeof item.openBlankWithCopyAccessCode === "boolean") {
                functionFormsList.push({
                  text: "跳转时复制访问码",
                  type: "switch",
                  description:
                    "作用于【新标签页打开】且存在访问码就复制到剪贴板",
                  attributes: {
                    "data-key": `${item.key}-open-blank-with-copy-accesscode`,
                    "data-default-value": false,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                  },
                });
              }
              /* 单文件解析|文件解析 */
              if (item.parseOneFile || item.parseMoreFile) {
                functionFormsList.push({
                  text: `${item.parseOneFile ? "单文件解析" : "文件解析"}`,
                  type: "switch",
                  description: item["parseFileDescription"] || "解析网盘链接",
                  attributes: {
                    "data-key": `${item.key}-static-enable`,
                    "data-default-value": false,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                    if (item.isBlank) {
                      let $shadowRoot = event.target.getRootNode();
                      /* 新标签页打开和单/文件解析只能开启一个 */
                      /**
                       * @type {HTMLInputElement}
                       */
                      let checkboxElement = $shadowRoot.querySelector(
                        `li[data-key="${item.key}-open-enable"] input[type=checkbox]`
                      );
                      let checkboxCoreElement = $shadowRoot.querySelector(
                        `li[data-key="${item.key}-open-enable"] span.pops-panel-switch__core`
                      );
                      if (value == true && checkboxElement.checked == true) {
                        checkboxCoreElement.click();
                      }
                    }
                  },
                });
              }

              /* 验证链接有效性 */
              if (typeof item.checkLinkValidity === "boolean") {
                functionFormsList.push({
                  text: `验证链接有效性`,
                  type: "switch",
                  description: "自动请求链接,判断该链接是否有效",
                  attributes: {
                    "data-key": `${item.key}-check-link-valid`,
                    "data-default-value": false,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                  },
                });
              }

              formsList.push({
                text: "功能",
                type: "forms",
                forms: functionFormsList,
              });
            }

            /* Scheme转发直链 */
            if (item.isForward) {
              /**
               * @type {PopsPanelFormsTotalDetails[]}
               */
              let functionFormsList = [
                {
                  text: "启用",
                  type: "switch",
                  description: "开启后可进行scheme uri转发",
                  attributes: {
                    "data-key": `${item.key}-forward-scheme-enable`,
                    "data-default-value": false,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                  },
                },
              ];
              if (typeof item.isForwardDownloadLink === "boolean") {
                functionFormsList.push({
                  text: "转发直链",
                  type: "switch",
                  description: "对解析的直链进行scheme转换",
                  attributes: {
                    "data-key": `${item.key}-forward-download-link-enable`,
                    "data-default-value": false,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                  },
                });
              }
              if (typeof item.isForwardBlankLink === "boolean") {
                functionFormsList.push({
                  text: "转发新标签页链接",
                  type: "switch",
                  description: "对新标签页打开的链接进行scheme转换",
                  attributes: {
                    "data-key": `${item.key}-forward-blank-link-enable`,
                    "data-default-value": false,
                  },
                  getValue() {
                    return Boolean(
                      GM_getValue(
                        this.attributes["data-key"],
                        this.attributes["data-default-value"]
                      )
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], Boolean(value));
                  },
                });
              }

              /* Scheme的Uri链接 */
              if ("schemeUri" in item) {
                functionFormsList.push({
                  text: "Uri链接",
                  type: "input",
                  description: "自定义的Scheme的Uri链接",
                  attributes: {
                    "data-key": `${item.key}-static-scheme-uri`,
                    "data-default-value": "",
                  },
                  getValue() {
                    return GM_getValue(
                      this.attributes["data-key"],
                      this.attributes["data-default-value"]
                    );
                  },
                  callback(event, value) {
                    GM_setValue(this.attributes["data-key"], value);
                  },
                  placeholder:
                    "jumpwsv://go?package=xx&activity=xx&intentAction=xx&intentData=xx&intentExtra=xx",
                });
              }
              formsList.push({
                text: "Scheme Uri转发",
                type: "forms",
                forms: functionFormsList,
              });
            }
            if (item.ownFormList) {
              formsList = formsList.concat(item.ownFormList);
            }

            let asideTitle = item.key;
            if (item.key in NetDiskUI.src.icon) {
              if (pops.isPhone()) {
                asideTitle = `
                <div style="
                    width: 20px;
                    height: 20px;
                    background: url(${NetDiskUI.src.icon[item.key]}) no-repeat;
                    background-size: 100% 100%;
                    ">`;
              } else {
                asideTitle = `
                <div style="
                    width: 20px;
                    height: 20px;
                    background: url(${NetDiskUI.src.icon[item.key]}) no-repeat;
                    background-size: 100% 100%;
                    "></div>
                <div style="margin-left: 4px;">${item.type}</div>`;
              }
            }
            let headerTitleText = item.type;
            if (item.isUserRule) {
              headerTitleText += `<div class="netdisk-custom-rule-edit" data-key="${item.key}" data-type="${item.type}">${pops.config.iconSVG.edit}</div>`;
              headerTitleText += `<div class="netdisk-custom-rule-delete" data-key="${item.key}" data-type="${item.type}">${pops.config.iconSVG.delete}</div>`;
            }
            contentDetails.push({
              id: "netdisk-panel-config-" + item.key,
              title: asideTitle,
              headerTitle: headerTitleText,
              attributes: {
                "data-key": item.key,
              },
              forms: formsList,
            });
          });
        }
        addNetDiskSetting();

        NetDiskUI.Alias.settingAlias = NetDiskPops.panel(
          {
            title: {
              text: `${GM_info?.script?.name || "网盘链接识别"}-设置`,
              position: "center",
            },
            content: contentDetails,
            btn: {
              close: {
                enable: true,
                callback(event) {
                  event.close();
                  NetDiskUI.Alias.settingAlias = void 0;
                },
              },
            },
            mask: {
              clickCallBack(originalRun) {
                originalRun();
                NetDiskUI.Alias.settingAlias = void 0;
              },
            },
            class: "whitesevPopSetting",
            style: `
            div[class^="netdisk-custom-rule-"]{
              display: flex;
              align-items: center;
              margin-left: 10px;
              cursor: pointer;
            }
            div[class^="netdisk-custom-rule-"] svg,
            div[class^="netdisk-custom-rule-"] svg{
              width: 1.2em;
              height: 1.2em;
            }
            `,
          },
          NetDiskUI.popsStyle.settingView
        );
        DOMUtils.on(
          NetDiskUI.Alias.settingAlias.$shadowRoot,
          "click",
          ".netdisk-custom-rule-edit",
          function (event) {
            let ruleKey = event.target.getAttribute("data-key");
            let ruleName = event.target.getAttribute("data-type");
            NetDiskCustomRules.showUI(true, ruleKey);
          }
        );

        DOMUtils.on(
          NetDiskUI.Alias.settingAlias.$shadowRoot,
          "click",
          ".netdisk-custom-rule-delete",
          function (event) {
            let ruleKey = event.target.getAttribute("data-key");
            let ruleName = event.target.getAttribute("data-type");
            NetDiskPops.alert({
              title: {
                text: "提示",
                position: "center",
              },
              content: {
                text: `确定删除自定义规则 ${ruleName}(${ruleKey}) 吗?`,
              },
              btn: {
                ok: {
                  callback(okEvent) {
                    let deleteStatus = NetDiskCustomRules.deleteRule(ruleKey);
                    if (deleteStatus) {
                      let asideElement =
                        NetDiskUI.Alias.settingAlias.$shadowRoot.querySelector(
                          `.pops-panel-aside > ul > li[data-key="${ruleKey}"]`
                        );
                      if (asideElement.previousElementSibling) {
                        asideElement.previousElementSibling.click();
                      } else if (asideElement.nextElementSibling) {
                        asideElement.nextElementSibling.click();
                      }
                      asideElement?.remove();
                      Qmsg.success("删除成功");
                      okEvent.close();
                    } else {
                      Qmsg.error("删除自定义规则失败");
                    }
                  },
                },
              },
            });
          }
        );
      },
      /**
       * 设置 悬浮按钮所有事件
       */
      setSuspensionEvent() {
        let that = this;
        let needDragElement = NetDiskUI.suspension.suspensionNode;
        let dragNode = new AnyTouch(needDragElement);
        /** @type {?number[]} */
        let netDiskLinkViewTimer = void 0;
        let moveFlag = false;
        /* 是否是双击 */
        let isDouble = false;
        /* 点击元素,left偏移 */
        let clickElementLeftOffset = 0;
        /* 点击元素,top偏移 */
        let clickElementTopOffset = 0;
        /* 设置悬浮按钮 按下事件 */
        dragNode.on("pan", function (event) {
          if (!moveFlag) {
            moveFlag = true;
            let rect = needDragElement.getBoundingClientRect();
            clickElementLeftOffset = event.x - rect.left;
            clickElementTopOffset = event.y - rect.top;
            DOMUtils.css(needDragElement, {
              cursor: "move",
              transition: "none",
            });
          }
          /**
           * 移动
           */
          if (event.phase === "move") {
            /* 悬浮按钮大小不能超过250px */
            /* left偏移最大值 */
            let maxLeftOffset = DOMUtils.width(globalThis) - NetDiskUI.size;
            /* top偏移的最大值 */
            let maxTopOffset = DOMUtils.height(globalThis) - NetDiskUI.size;
            /* 当前移动的left偏移 */
            let currentSuspensionLeftOffset = event.x - clickElementLeftOffset;
            /* 当前移动的top偏移 */
            let currentSuspensionTopOffset = event.y - clickElementTopOffset;
            /* 不允许超过窗口最大宽度 */
            currentSuspensionLeftOffset =
              currentSuspensionLeftOffset > maxLeftOffset
                ? maxLeftOffset
                : currentSuspensionLeftOffset;
            /* 不允许超过窗口最大高度 */
            currentSuspensionTopOffset =
              currentSuspensionTopOffset > maxTopOffset
                ? maxTopOffset
                : currentSuspensionTopOffset;
            /* 不允许小于0 */
            currentSuspensionLeftOffset =
              currentSuspensionLeftOffset < 0 ? 0 : currentSuspensionLeftOffset;
            /* 不允许小于0 */
            currentSuspensionTopOffset =
              currentSuspensionTopOffset < 0 ? 0 : currentSuspensionTopOffset;
            if (NetDiskUI.suspension.isTopWindow()) {
              GM_setValue("suspensionX", currentSuspensionLeftOffset);
              GM_setValue("suspensionY", currentSuspensionTopOffset);
            }
            DOMUtils.css(needDragElement, {
              left: currentSuspensionLeftOffset + "px",
              top: currentSuspensionTopOffset + "px",
            });
          }

          /**
           * 停止移动
           */
          if (event.phase === "end") {
            moveFlag = false;
            DOMUtils.css(needDragElement, {
              cursor: "auto",
            });
            /* 获取当前悬浮按钮left偏移 */
            let currentSuspensionLeftOffset = parseInt(
              DOMUtils.css(needDragElement, "left")
            );
            if (GM_getValue("suspended-button-adsorption-edge")) {
              let setCSSLeft = 0;
              /* 判断悬浮按钮是否在右边区域 */
              if (
                currentSuspensionLeftOffset >=
                DOMUtils.width(globalThis) / 2
              ) {
                /* 设置悬浮按钮的left偏移 */
                setCSSLeft = DOMUtils.width(globalThis) - NetDiskUI.size;
                if (NetDiskUI.suspension.isTopWindow()) {
                  GM_setValue("isRight", true);
                }
              } else {
                if (NetDiskUI.suspension.isTopWindow()) {
                  GM_setValue("isRight", false);
                }
              }
              if (NetDiskUI.suspension.isTopWindow()) {
                GM_setValue("suspensionX", setCSSLeft);
              }
              DOMUtils.css(needDragElement, {
                left: setCSSLeft + "px",
              });
            }
            DOMUtils.css(needDragElement, {
              transition: "left 300ms ease 0s",
            });
          }
        });
        /* 设置悬浮按钮 点击/按下事件 */
        dragNode.on(["click", "tap"], function (event) {
          clearTimeout(netDiskLinkViewTimer);
          netDiskLinkViewTimer = void 0;
          if (isDouble) {
            isDouble = false;
            /* 判定为双击 */
            NetDiskUI.suspension.showSettingView();
          } else {
            netDiskLinkViewTimer = setTimeout(() => {
              isDouble = false;
              if (
                NetDiskUIMenuData.netdiskBehaviorMode.includes("smallwindow")
              ) {
                GM_setValue(
                  "current_suspension_smallwindow_mode",
                  "smallwindow"
                );
                NetDiskUI.suspension.hideSuspension();
              }
              NetDiskUI.view.show();
            }, 200);
            isDouble = true;
          }
        });
        NetDiskUI.setGlobalRightClickMenu(needDragElement);
      },
      /**
       * 设置window的resize事件监听,来重新设置悬浮按钮的位置
       */
      setResizeEventListener() {
        DOMUtils.on(globalThis, "resize", void 0, () => {
          let activeElement = document.activeElement;
          if (utils.isPhone()) {
            if (["input", "textarea"].includes(activeElement.localName)) {
              /* 可能是移动端的输入框弹出的键盘导致的resize */
              return;
            } else if (
              (activeElement.hasAttribute("contenteditable") &&
                activeElement.getAttribute("contenteditable") === "true") ||
              activeElement.closest("[contenteditable='true']")
            ) {
              /* 可能是移动端的输入框弹出的键盘导致的resize */
              return;
            } else if (!document.hasFocus()) {
              /* 页面失焦 */
              return;
            }
          }
          this.setSuspensionPosition();
        });
      },
      /**
       * 设置悬浮按钮位置
       */
      setSuspensionPosition() {
        /* 最大的left偏移*/
        let maxLeftOffset = DOMUtils.width(globalThis) - NetDiskUI.size;
        /* 最大的top偏移 */
        let maxTopOffset = DOMUtils.height(globalThis) - NetDiskUI.size;
        /* 默认的left偏移(悬浮按钮在右边中间) */
        let defaultLeftOffset = maxLeftOffset;
        /* 默认的top偏移(悬浮按钮在右边中间) */
        let defaultTopOffset = maxTopOffset / 2;
        /* 用户自己拖动设置的悬浮按钮left偏移 */
        let userSetLeftOffset = GM_getValue("suspensionX", defaultLeftOffset);

        /* 用户自己拖动设置的悬浮按钮top偏移 */
        let userSetTopOffset = GM_getValue("suspensionY", defaultTopOffset);

        if (GM_getValue("suspended-button-adsorption-edge")) {
          /* 如果isRight为true,悬浮按钮放到最右边,否则最左边 */
          if (GM_getValue("isRight")) {
            userSetLeftOffset = maxLeftOffset;
          } else {
            userSetLeftOffset = 0;
          }
          /* 如果用户设置的top偏移超出最大的top偏移,那么设置用户的偏移为默认的最大top偏移 */
          if (userSetTopOffset > maxTopOffset) {
            userSetTopOffset = maxTopOffset;
          } else if (userSetTopOffset < 0) {
            /* 如果用户设置的top偏移为负的,那么是超出边界,归位设置为0 */
            userSetTopOffset = 0;
          }

          if (NetDiskUI.suspension.isTopWindow()) {
            /* 当前窗口是顶部窗口,才可以保存移动的值 */
            GM_setValue("suspensionX", userSetLeftOffset);
            GM_setValue("suspensionY", userSetTopOffset);
          }
        }
        DOMUtils.css(NetDiskUI.suspension.suspensionNode, {
          left: userSetLeftOffset + "px",
          top: userSetTopOffset + "px",
        });
      },
      getCSS() {
        return `
					.whitesevSuspension{
						top: 0;
						position:fixed;
						right:10px;
						border-radius: 12px;
						z-index:4000;
					}
					.whitesevSuspension .whitesevSuspensionMain{
						background:#fff;
						border:1px solid #f2f2f2;
						box-shadow:0 0 15px #e4e4e4;
						box-sizing:border-box;
						border-radius: inherit;
						height: inherit;
						width: inherit;
					}
					.whitesevSuspension .whitesevSuspensionFloor{
						border-bottom:1px solid #f2f2f2;
						position:relative;
						box-sizing:border-box;
						border-radius: inherit;
						height: inherit;
						width: inherit;
					}
					.whitesevSuspension .whitesevSuspensionFloor .netdisk{
						background-position:center center;
						background-size:115% 115%;
						background-repeat:no-repeat;
						display:flex;
						align-items:center;
						justify-content:center;
						border-radius: inherit;
						height: inherit;
						width: inherit;
					}
					.whitesevSuspension .whitesevSuspensionFloor .netdisk:hover{
						transition:all 300ms linear;
						background-color:#e4e4e4;
						transform:scale(1.1);
					}
					.whitesevPop-content p[pop]{
						height: 100%;
					}
				`;
      },
      /**
       * 悬浮按钮背景轮播 效果为淡入淡出
       */
      backgroundSwitch() {
        if (this.isRandBg) {
          return;
        }
        /**
         * 获取随机背景的数组
         * @returns {Array}
         */
        function getRandBgList() {
          let resultList = [];
          NetDiskUI.isMatchedNetDiskIconMap.forEach((item) => {
            resultList = [...resultList, NetDiskUI.src.icon[item]];
          });
          return resultList;
        }
        /**
         * 进行切换 淡入淡出
         * @param {number} fadeTime 淡入\淡出的时间
         * @param {string} currentBackgroundSrc 当前的背景资源
         */
        function startSwitch(fadeTime, currentBackgroundSrc) {
          currentList = getRandBgList();
          DOMUtils.fadeOut(randBgNode, fadeTime, function () {
            currentIndex++;
            currentIndex = currentIndex < currentList.length ? currentIndex : 0;
            currentBackgroundSrc = currentList[currentIndex];
            DOMUtils.css(
              randBgNode,
              "background-image",
              `url(${currentBackgroundSrc})`
            );
            DOMUtils.fadeIn(randBgNode, fadeTime, function () {
              setTimeout(() => {
                startSwitch(
                  parseInt(GM_getValue("randbg-time", switchBgTime)),
                  currentBackgroundSrc
                );
              }, parseInt(GM_getValue("randbg-show-time", switchBgShowTime)));
            });
          });
        }
        let currentList = [];
        let currentIndex = 0;
        let switchBgTime = 1500; /* 淡入或淡出的持续时间 */
        let switchBgShowTime = 1200; /* 淡入或淡出后的延迟切换时间 */
        currentList = getRandBgList();
        let randBgSrc = currentList[currentIndex];
        let randBgNode = NetDiskUI.suspension.suspensionNode.querySelector(
          ".whitesevSuspension .netdisk"
        );
        DOMUtils.css(randBgNode, "background-image", `url(${randBgSrc})`);
        if (
          currentList.length < 2 ||
          GM_getValue("randbg-time", switchBgTime) <= 0
        ) {
          /* 只有一个的话或淡入/淡出的时间<=0 就不进行背景切换 */
          return;
        }
        this.isRandBg = true;

        startSwitch(
          parseInt(GM_getValue("randbg-time", switchBgTime)),
          randBgSrc
        );
      },
    },
    /**
     * 主视图
     */
    view: {
      show() {
        if (!NetDiskUI.Alias.uiLinkAlias) {
          this.createView();
          NetDiskUI.setRightClickMenu(
            NetDiskUI.Alias.uiLinkAlias.$shadowRoot,
            ".whitesevPop .netdisk-url a"
          );
          this.registerIconGotoPagePosition(
            NetDiskUI.Alias.uiLinkAlias.$shadowRoot
          );
          this.setNetDiskUrlClickEvent(
            NetDiskUI.Alias.uiLinkAlias.$shadowRoot,
            ".netdisk-url a"
          );
          NetDiskUI.setGlobalRightClickMenu(
            NetDiskUI.Alias.uiLinkAlias.$shadowRoot.querySelector(
              ".pops .pops-alert-title > p"
            )
          );
        } else {
          NetDiskUI.Alias.uiLinkAlias.show();
        }
      },
      getCSS() {
        return `
        .netdisk-url-box{
          border-bottom: 1px solid #e4e6eb;
        }
        .netdisk-url-div{display:flex;align-items:center;width:100%;padding:5px 0px 5px 0px}
        .netdisk-icon{display:contents}
        .netdisk-icon .netdisk-icon-img{
          cursor: pointer;
          width:28px;
          height:28px;
          min-width:28px;
          min-height:28px;
          font-size:0.8em;
          margin: 0px 10px;
        }
        .netdisk-url-div .netdisk-icon,
        .netdisk-url-div .netdisk-status{
          flex: 0 0 auto;
        }
        .netdisk-url-div .netdisk-url{
          flex: 1;
        }
        .netdisk-icon .netdisk-icon-img{
          border-radius: 10px;
          box-shadow: 0 .3px .6px rgb(0 0 0 / 6%),0 .7px 1.3px rgb(0 0 0 / 8%),0 1.3px 2.5px rgb(0 0 0 / 10%),0 2.2px 4.5px rgb(0 0 0 / 12%),0 4.2px 8.4px rgb(0 0 0 / 14%),0 10px 20px rgb(0 0 0 / 20%)
        }
        .netdisk-status[data-check-failed]{
          padding: 5px 5px;
        }
        .netdisk-url{padding:5px 5px;}
        .netdisk-url a {
          color: #ff4848!important;
          min-height: 28px;
          overflow-x: hidden;
          overflow-y: auto;
          font-size: 0.8em;
          border: none;
          display: flex;
          align-items: center;
          width: 100%;
          height: 100%;
          padding: 0px;
          word-break: break-word;
          text-align: left;
        }
        .netdisk-status{
          display: none;
        }
        .netdisk-status[data-check-valid]{
          display: flex;
          align-items: center;
          width: 15px;
          height: 15px;
        }
        .netdisk-status[data-check-valid="failed"]{
          color: red;
        }
        .netdisk-status[data-check-valid="error"]{
          cursor: pointer;
        }
        .netdisk-status[data-check-valid="success"]{
          color: green;
        }
        .netdisk-status[data-check-valid="loading"] svg{
          animation: rotating 2s linear infinite;
        }
        .netdisk-url-box:has(.netdisk-status[data-check-valid="failed"]){
          text-decoration: line-through;
        }
        .whitesevPop-whitesevPopSetting :focus-visible{outline-offset:0;outline:0}
        .netdisk-url a[isvisited=true]{color:#8b8888!important}
        .netdisk-url a:active{box-shadow:0 0 0 1px #616161 inset}
        .netdisk-url a:focus-visible{outline:0}
        .whitesevPop-content p[pop]{text-indent:0}
        .whitesevPop-button[type=primary]{border-color:#2d8cf0;background-color:#2d8cf0}
				`;
      },
      /**
       * 创建视图
       */
      createView() {
        let viewAddHTML = "";
        NetDiskUI.isMatchedNetDiskIconMap.forEach((netDiskName) => {
          let netDiskDict = NetDisk.linkDict.get(netDiskName);
          let netDiskData = netDiskDict.getItems();
          Object.keys(netDiskData).forEach((shareCode) => {
            let accessCodeDict = netDiskData[shareCode];
            let uiLink = NetDisk.handleLinkShow(
              netDiskName,
              accessCodeDict["netDiskIndex"],
              shareCode,
              accessCodeDict["accessCode"],
              accessCodeDict["matchText"]
            );
            viewAddHTML =
              viewAddHTML +
              this.getViewHTML(
                NetDiskUI.src.icon[netDiskName],
                netDiskName,
                accessCodeDict["netDiskIndex"],
                shareCode,
                accessCodeDict["accessCode"],
                uiLink
              );
          });
        });
        let viewHTML = `
					<div class="netdisk-url-box-all">
						${viewAddHTML}
					</div>`;

        if (
          NetDiskUIMenuData.netdiskBehaviorMode
            .toLowerCase()
            .includes("smallwindow")
        ) {
          NetDiskUI.Alias.uiLinkAlias = NetDiskPops.alert(
            {
              title: {
                text: "网盘",
                position: "center",
              },
              content: {
                text: viewHTML,
                html: true,
              },
              btn: {
                ok: {
                  enable: false,
                },
                close: {
                  callback(event) {
                    if (
                      NetDiskUIMenuData.netdiskBehaviorMode
                        .toLowerCase()
                        .includes("suspension")
                    ) {
                      GM_setValue(
                        "current_suspension_smallwindow_mode",
                        "suspension"
                      );
                      event.hide();
                      NetDiskUI.suspension.show();
                    } else {
                      NetDiskUI.Alias.uiLinkAlias = void 0;
                      event.close();
                    }
                  },
                },
              },
              mask: {
                enable: false,
              },
              animation: "",
              beforeAppendToPageCallBack($shadowRoot, $shadowContainer) {
                let buttonHeaderControl = $shadowRoot.querySelector(
                  ".pops-header-control"
                );
                let alertContent = $shadowRoot.querySelector(
                  ".pops-alert-content"
                );
                /* 展开 */
                let launchIcon = DOMUtils.createElement(
                  "button",
                  {
                    className: "pops-header-control",
                    innerHTML: `
                    <i class="pops-icon">
                      <svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg">
                        <path fill="currentColor" d="M290.816 774.144h167.936c12.288 0 20.48 8.192 20.48 20.48s-8.192 20.48-20.48 20.48h-219.136c-12.288 0-20.48-8.192-20.48-20.48v-2.048-206.848c0-12.288 8.192-20.48 20.48-20.48s20.48 8.192 20.48 20.48v163.84l210.944-198.656c8.192-8.192 20.48-8.192 28.672 0s8.192 20.48 0 28.672l-208.896 194.56z m462.848-524.288h-167.936c-12.288 0-20.48-8.192-20.48-20.48s8.192-20.48 20.48-20.48h219.136c12.288 0 20.48 8.192 20.48 20.48v208.896c0 12.288-8.192 20.48-20.48 20.48s-20.48-8.192-20.48-20.48v-163.84l-210.944 198.656c-8.192 8.192-20.48 8.192-28.672 0s-8.192-20.48 0-28.672l208.896-194.56z m188.416 323.584c0 12.288-8.192 20.48-20.48 20.48s-20.48-8.192-20.48-20.48v-389.12c0-34.816-26.624-61.44-61.44-61.44h-655.36c-34.816 0-61.44 26.624-61.44 61.44v655.36c0 34.816 26.624 61.44 61.44 61.44h655.36c34.816 0 61.44-26.624 61.44-61.44v-94.208c0-12.288 8.192-20.48 20.48-20.48s20.48 8.192 20.48 20.48v94.208c0 57.344-45.056 102.4-102.4 102.4h-655.36c-57.344 0-102.4-45.056-102.4-102.4v-655.36c0-57.344 45.056-102.4 102.4-102.4h655.36c57.344 0 102.4 45.056 102.4 102.4v389.12z">
                        </path>
                      </svg>
                    </i>
                  `,
                  },
                  {
                    type: "launch",
                    "data-header": true,
                  }
                );
                /* 收起 */
                let shrinkIcon = DOMUtils.createElement(
                  "button",
                  {
                    className: "pops-header-control",
                    innerHTML: `
                    <i class="pops-icon">
                      <svg viewBox="0 0 1024 1024" xmlns="http://www.w3.org/2000/svg">
                        <path fill="currentColor" d="M618.496 425.984h167.936c12.288 0 20.48 8.192 20.48 20.48s-8.192 20.48-20.48 20.48h-219.136c-12.288 0-20.48-8.192-20.48-20.48v-2.048-206.848c0-12.288 8.192-20.48 20.48-20.48s20.48 8.192 20.48 20.48v163.84l210.944-198.656c8.192-8.192 20.48-8.192 28.672 0s8.192 20.48 0 28.672l-208.896 194.56z m-192.512 172.032h-167.936c-12.288 0-20.48-8.192-20.48-20.48s8.192-20.48 20.48-20.48h219.136c12.288 0 20.48 8.192 20.48 20.48v208.896c0 12.288-8.192 20.48-20.48 20.48s-20.48-8.192-20.48-20.48v-163.84l-210.944 198.656c-8.192 8.192-20.48 8.192-28.672 0s-8.192-20.48 0-28.672l208.896-194.56z m516.096-24.576c0 12.288-8.192 20.48-20.48 20.48s-20.48-8.192-20.48-20.48v-389.12c0-34.816-26.624-61.44-61.44-61.44h-655.36c-34.816 0-61.44 26.624-61.44 61.44v655.36c0 34.816 26.624 61.44 61.44 61.44h655.36c34.816 0 61.44-26.624 61.44-61.44v-94.208c0-12.288 8.192-20.48 20.48-20.48s20.48 8.192 20.48 20.48v94.208c0 57.344-45.056 102.4-102.4 102.4h-655.36c-57.344 0-102.4-45.056-102.4-102.4v-655.36c0-57.344 45.056-102.4 102.4-102.4h655.36c57.344 0 102.4 45.056 102.4 102.4v389.12z">
                        </path>
                      </svg>
                    </i>
                  `,
                  },
                  {
                    type: "shrink",
                    "data-header": true,
                  }
                );
                DOMUtils.before(buttonHeaderControl, launchIcon);
                DOMUtils.before(buttonHeaderControl, shrinkIcon);
                DOMUtils.on(
                  launchIcon,
                  "click",
                  void 0,
                  function () {
                    /* 展开-切换为收缩图标 */
                    launchIcon.classList.add("pops-hide-important");
                    shrinkIcon.classList.remove("pops-hide-important");
                    alertContent.classList.remove("pops-hide-important");
                    GM_setValue("netdisl-small-window-shrink-status", false);
                  },
                  {
                    capture: true,
                  }
                );
                DOMUtils.on(
                  shrinkIcon,
                  "click",
                  void 0,
                  function () {
                    /* 收缩-切换为展开图标 */
                    shrinkIcon.classList.add("pops-hide-important");
                    launchIcon.classList.remove("pops-hide-important");
                    alertContent.classList.add("pops-hide-important");
                    alertContent.classList.add("pops-no-border-important");
                    GM_setValue("netdisl-small-window-shrink-status", true);
                  },
                  {
                    capture: true,
                  }
                );
                if (GM_getValue("netdisl-small-window-shrink-status", false)) {
                  shrinkIcon.click();
                } else {
                  launchIcon.click();
                }
              },
              dragMoveCallBack(moveElement, left, top) {
                GM_setValue("netdisk-ui-small-window-position", {
                  left: left,
                  top: top,
                });
              },
              class: "whitesevPop",
              style: `
              ${this.getCSS()}

              .pops {
                --container-title-height: 35px;
                --content-max-height: ${GM_getValue(
                  "netdisk-ui-small-window-max-height",
                  NetDiskData.defaultSmallWindowMaxHeight
                )}px;
                --netdisk-line-space: 8px;
                --netdisk-icon-size: 24px;
              }
              .pops[type-value="alert"]{
                transform: none;
              }
              .pops {
                max-height: var(--content-max-height);
              }
              .pops[type-value=alert] .pops-alert-content{
                max-height: calc(var(--content-max-height) - var(--container-title-height) - var(--container-bottom-btn-height));
              }
              .pops-header-controls button.pops-header-control[type][data-header]{
                padding: 0px 5px;
              }
              .netdisk-url-div{
                padding: 0px;
              }
              .netdisk-icon .netdisk-icon-img{
                width: var(--netdisk-icon-size);
                height: var(--netdisk-icon-size);
                min-width: var(--netdisk-icon-size);
                min-height: var(--netdisk-icon-size);
                margin: 0px var(--netdisk-line-space);
              }
              .netdisk-status{
                margin-right: var(--netdisk-line-space);
              }
              .netdisk-url{
                padding: 2px 0px;
              }
              `,
            },
            NetDiskUI.popsStyle.mainViewSmallWindow
          );

          let smallWindowPosition = GM_getValue(
            "netdisk-ui-small-window-position"
          );
          let popsElement = NetDiskUI.Alias.uiLinkAlias.popsElement;
          if (smallWindowPosition) {
            let viewWidth = DOMUtils.width(popsElement, true);
            let viewHeight = DOMUtils.height(popsElement, true);
            let maxWindowLeft = DOMUtils.width(globalThis);
            let maxWindowTop = DOMUtils.height(globalThis);
            const { transformLeft, transformTop } =
              DOMUtils.getTransform(popsElement);
            /* 最大的left偏移*/
            let maxLeft = maxWindowLeft - viewWidth + transformLeft;
            /* 最大的top偏移 */
            let maxTop = maxWindowTop - viewHeight + transformTop;
            /* 最小的left偏移*/
            let minLeft = 0 + transformLeft;
            /* 最小的top偏移*/
            let minTop = 0 + transformTop;
            if (smallWindowPosition.top > maxTop) {
              smallWindowPosition.top = maxTop;
            } else if (smallWindowPosition.top < minTop) {
              smallWindowPosition.top = minTop;
            }
            if (smallWindowPosition.left > maxLeft) {
              smallWindowPosition.left = maxLeft;
            } else if (smallWindowPosition.left < minLeft) {
              smallWindowPosition.left = minLeft;
            }
            popsElement.style.transitionDuration = "0s";
            popsElement.style.left = smallWindowPosition["left"] + "px";
            popsElement.style.top = smallWindowPosition["top"] + "px";
            setTimeout(() => {
              popsElement.style.transitionDuration = "0s";
            }, 300);
          }
        } else {
          NetDiskUI.Alias.uiLinkAlias = NetDiskPops.alert(
            {
              title: {
                text: "网盘",
                position: "center",
              },
              content: {
                text: viewHTML,
                html: true,
              },
              btn: {
                ok: {
                  enable: false,
                },
                close: {
                  callback(event) {
                    NetDiskUI.Alias.uiLinkAlias = void 0;
                    event.close();
                  },
                },
              },
              mask: {
                clickCallBack(originalRun) {
                  originalRun();
                  NetDiskUI.Alias.uiLinkAlias = void 0;
                },
              },
              class: "whitesevPop",
              style: `
              ${this.getCSS()}

              .pops {
                max-height: ${pops.isPhone() ? "50dvh" : "60dvh"};
              }
              `,
            },
            NetDiskUI.popsStyle.mainView
          );
        }

        NetDiskUI.Alias.uiLinkAlias.popsElement
          .querySelectorAll(".netdisk-url-box-all .netdisk-url-box")
          .forEach((ele) => {
            let netDiskName = ele
              .querySelector(".netdisk-link")
              .getAttribute("data-netdisk");
            let netDiskIndex = parseInt(
              ele
                .querySelector(".netdisk-link")
                .getAttribute("data-netdisk-index")
            );
            let shareCode = ele
              .querySelector(".netdisk-link")
              .getAttribute("data-sharecode");
            let accessCode = ele
              .querySelector(".netdisk-link")
              .getAttribute("data-accesscode");
            NetDiskCheckLinkValidity.check(
              ele,
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
          });
      },
      /**
       * 获取视图html
       * @param {string} netDiskImgSrc 网盘图标src
       * @param {string} netDiskName 网盘名称
       * @param {number} netDiskIndex 网盘名称索引下标
       * @param {string} shareCode
       * @param {string} accessCode
       * @param {string} uiLinkText 显示出来的链接文本
       * @returns {string}
       */
      getViewHTML(
        netDiskImgSrc,
        netDiskName,
        netDiskIndex,
        shareCode,
        accessCode,
        uiLinkText
      ) {
        return `
				<div class="netdisk-url-box">
					<div class="netdisk-url-div">
						<div class="netdisk-icon">
							<div class="netdisk-icon-img"
                  style="background: url(${netDiskImgSrc}) no-repeat;background-size: 100%;"
                  data-netdisk="${netDiskName}"
                  data-sharecode="${shareCode}"
                  data-accesscode="${accessCode}">
						  </div>
            </div>
            <div class="netdisk-status">

            </div>
						<div class="netdisk-url">
							<a  class="netdisk-link"
                  href="javascript:;" 
                  isvisited="false"
                  data-netdisk="${netDiskName}"
                  data-netdisk-index="${netDiskIndex}"
                  data-sharecode="${shareCode}"
                  data-accesscode="${accessCode}"
                  data-open-enable-key="${netDiskName}-open-enable"
                  data-static-enable-key="${netDiskName}-static-enable"
                  data-forward-scheme-key="${netDiskName}-forward-scheme-enable">${uiLinkText}</a>
						</div>
					</div>
				</div>`;
      },
      /**
       * 设置网盘链接点击事件,要求,它是<a>元素且存在以下属性
       * isvisited  string true|false
       * data-netdisk  string
       * data-netdisk-index number
       * data-sharecode string
       * data-accesscode string
       * data-open-enable-key string
       * data-static-enable-key string
       * data-forward-scheme-key string
       * @param {*} target
       * @param {string} clickNodeSelector - 元素选择器
       */
      setNetDiskUrlClickEvent(target, clickNodeSelector) {
        function clickEvent(event) {
          event.target.setAttribute("isvisited", "true");
          let netDiskName = event.target.getAttribute("data-netdisk");
          let netDiskIndex = parseInt(
            event.target.getAttribute("data-netdisk-index")
          );
          let shareCode = event.target.getAttribute("data-sharecode");
          let accessCode = event.target.getAttribute("data-accesscode");
          let openEnable = GM_getValue(
            event.target.getAttribute("data-open-enable-key"),
            false
          );
          let staticEnable = GM_getValue(
            event.target.getAttribute("data-static-enable-key"),
            false
          );
          let schemeEnable = GM_getValue(
            event.target.getAttribute("data-forward-scheme-key"),
            false
          );
          if (openEnable) {
            let url = NetDiskParse.getBlankUrl(
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
            NetDiskParse.blank(
              url,
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
          } else if (staticEnable) {
            NetDiskParse.parse(
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
          } else if (schemeEnable) {
            NetDiskParse.openScheme(
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
          } else {
            NetDiskParse.copyText(
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
          }
        }
        DOMUtils.on(target, "click", clickNodeSelector, clickEvent);
      },
      /**
       * 注册右键菜单
       * @param {HTMLElement|Window} target
       * @param {?string} selector
       * @param {{text:string,callback:Function}[]} showTextList 右键菜单的内容
       * @param {string} className className属性
       */
      registerContextMenu(
        target,
        selector,
        showTextList = [],
        className = "whitesevSuspensionContextMenu"
      ) {
        let data = [];
        showTextList.forEach((item) => {
          data.push({
            icon: "",
            text: item.text,
            callback: item.callback,
          });
        });
        pops.rightClickMenu({
          target: target,
          targetSelector: selector,
          data: data,
          isAnimation: false,
          className: className,
        });
      },
      /**
       * 添加新的链接
       * @param {string} netDiskName 网盘名称
       * @param {number} netDiskIndex 网盘名称索引下标
       * @param {string} shareCode 分享码
       * @param {string} accessCode 访问码
       * @param {string} matchText 匹配到的文本
       */
      addLinkView(netDiskName, netDiskIndex, shareCode, accessCode, matchText) {
        NetDiskUI.netDiskHistoryMatch.setNetDiskHistoryMatchData(
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode,
          matchText
        );
        if (!NetDiskUI.Alias.uiLinkAlias) {
          return;
        }
        log.info([netDiskName, netDiskIndex, shareCode, accessCode]);
        let icon = NetDiskUI.src.icon[netDiskName];
        let uiLink = NetDisk.handleLinkShow(
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode,
          matchText
        );
        let insertDOM = this.getViewHTML(
          icon,
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode,
          uiLink
        );
        DOMUtils.append(
          NetDiskUI.Alias.uiLinkAlias.popsElement.querySelector(
            ".netdisk-url-box-all"
          ),
          insertDOM
        );
        NetDiskCheckLinkValidity.check(
          NetDiskUI.Alias.uiLinkAlias.popsElement.querySelector(
            ".netdisk-url-box-all"
          ).children[
            NetDiskUI.Alias.uiLinkAlias.popsElement.querySelector(
              ".netdisk-url-box-all"
            ).children.length - 1
          ],
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode
        );
      },
      /**
       * 修改已存在的view
       * @param {string} netDiskName 网盘名称
       * @param {number} netDiskIndex 网盘名称索引下标
       * @param {string} shareCode 分享码
       * @param {string} accessCode 访问码
       * @param {string} matchText 匹配到的文本
       */
      changeLinkView(
        netDiskName,
        netDiskIndex,
        shareCode,
        accessCode,
        matchText
      ) {
        NetDiskUI.netDiskHistoryMatch.setNetDiskHistoryMatchData(
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode,
          matchText
        );
        if (!NetDiskUI.Alias.uiLinkAlias) {
          return;
        }
        let uiLink = NetDisk.handleLinkShow(
          netDiskName,
          netDiskIndex,
          shareCode,
          accessCode,
          matchText
        );
        let needChangeDOM =
          NetDiskUI.Alias.uiLinkAlias.popsElement.querySelector(
            `.netdisk-url a[data-sharecode='${shareCode}'][data-netdisk-index='${netDiskIndex}']`
          );
        log.info("修改网盘链接视图");
        log.info(needChangeDOM);
        needChangeDOM.setAttribute("data-accesscode", accessCode);
        DOMUtils.html(needChangeDOM, uiLink);
      },
      /**
       * 设置点击图标按钮导航至该网盘链接所在网页中位置
       */
      registerIconGotoPagePosition(target) {
        /**
         * @type {?Generator<HTMLElement, void, any>}
         */
        let findGenerator = void 0;
        /**
         * @type {?IteratorResult<HTMLElement, void>}
         */
        let iterator = void 0;
        /**
         * 上一个的shareCode
         */
        let prevSearchShareCode = void 0;
        DOMUtils.on(
          target,
          "click",
          ".whitesevPop .netdisk-icon .netdisk-icon-img",
          function (event) {
            let dataSharecode = event.target.getAttribute("data-sharecode");
            if (
              !GM_getValue(
                "pops-netdisk-icon-click-event-find-sharecode",
                NetDiskData.iconDefaultClickEventToFindShareCode
              )
            ) {
              return;
            }
            if (typeof dataSharecode !== "string") {
              Qmsg.error("获取data-sharecode属性失败");
              return;
            }
            if (prevSearchShareCode == void 0) {
              prevSearchShareCode = dataSharecode;
            } else if (prevSearchShareCode !== dataSharecode) {
              /* 切换到另一个shareCode搜索 */
              log.info(
                `上一个搜索:${prevSearchShareCode},切换至:${dataSharecode}`
              );
              findGenerator = void 0;
              iterator = void 0;
              prevSearchShareCode = dataSharecode;
            }
            if (findGenerator == void 0) {
              /* 未找到元素或者已迭代完毕 */
              findGenerator = utils.findElementsWithText(
                document.documentElement,
                dataSharecode
              );
              iterator = findGenerator.next();
            }
            if (iterator.value) {
              log.success(["定位元素", iterator]);
              if (
                iterator.value.nodeType === Node.ELEMENT_NODE &&
                iterator.value.getClientRects().length
              ) {
                /* 可见 */
                /* 滚动到该元素 */
                iterator.value.scrollIntoView({
                  behavior: "smooth",
                  block: "center",
                  inline: "nearest",
                });
                if (
                  GM_getValue(
                    "pops-netdisk-icon-click-event-find-sharecode-with-select",
                    NetDiskData.iconDefaultClickEventToFindShareCodeWithSelect
                  )
                ) {
                  /* 开启功能 */
                  let elementText =
                    iterator.value.innerText || iterator.value.textContent;
                  let childTextNode = void 0;
                  let startIndex = void 0;
                  let endIndex = void 0;
                  if (elementText.includes(dataSharecode)) {
                    /* 文字包含shareCode */
                    let textNodeList = Array.from(
                      iterator.value.childNodes
                    ).filter((ele) => ele.nodeType === Node.TEXT_NODE);
                    for (const textNode of textNodeList) {
                      if (textNode.textContent.includes(dataSharecode)) {
                        childTextNode = textNode;
                        startIndex =
                          textNode.textContent.indexOf(dataSharecode);
                        endIndex = startIndex + dataSharecode.length;
                        break;
                      }
                    }
                  }
                  try {
                    utils.selectElementText(
                      iterator.value,
                      childTextNode,
                      startIndex,
                      endIndex
                    );
                  } catch (error) {
                    log.error(error);
                    utils.selectElementText(iterator.value);
                  }
                }
              } else if (
                iterator.value.nodeType === Node.TEXT_NODE &&
                iterator.value.parentElement.getClientRects().length
              ) {
                /* #text元素且可见 */
                if (
                  GM_getValue(
                    "pops-netdisk-icon-click-event-find-sharecode-with-select",
                    NetDiskData.iconDefaultClickEventToFindShareCodeWithSelect
                  )
                ) {
                  let elementText =
                    iterator.value.textContent || iterator.value.nodeValue;
                  let childTextNode = iterator.value;
                  let startIndex = elementText.indexOf(dataSharecode);
                  let endIndex = startIndex + dataSharecode.length;
                  try {
                    utils.selectElementText(
                      iterator.value,
                      childTextNode,
                      startIndex,
                      endIndex
                    );
                  } catch (error) {
                    log.error(error);
                    utils.selectElementText(iterator.value.parentElement);
                  }
                  let selection = globalThis.getSelection();
                  if (selection.rangeCount > 0) {
                    let range = selection.getRangeAt(0);
                    let rect = range.getBoundingClientRect();
                    let scrollYOffset = globalThis.scrollY;
                    /* 居中定位 */
                    let position =
                      rect.top + scrollYOffset - globalThis.innerHeight / 2;
                    globalThis.scrollTo({
                      behavior: "smooth",
                      top: position,
                    });
                  } else {
                    iterator.value.parentElement.scrollIntoView({
                      behavior: "smooth",
                      block: "center",
                      inline: "nearest",
                    });
                  }
                } else {
                  try {
                    let range = new Range();
                    range.selectNodeContents(iterator.value);
                    let rect = range.getBoundingClientRect();
                    let scrollYOffset = globalThis.scrollY;
                    /* 居中定位 */
                    let position =
                      rect.top + scrollYOffset - globalThis.innerHeight / 2;
                    globalThis.scrollTo({
                      behavior: "smooth",
                      top: position,
                    });
                  } catch (error) {
                    log.error(error);
                    iterator.value.parentElement.scrollIntoView({
                      behavior: "smooth",
                      block: "center",
                      inline: "nearest",
                    });
                  }
                }
              } else {
                log.error(["无法定位该元素位置", iterator.value]);
                Qmsg.error(
                  `无法定位该元素位置,类型:<${(
                    iterator.value.nodeName ||
                    iterator.value.localName ||
                    iterator.value.tagName
                  ).toLowerCase()}>`,
                  {
                    html: false,
                  }
                );
              }
            }

            iterator = findGenerator.next();
            if (iterator.done) {
              /* 循环查找 */
              if (
                !GM_getValue(
                  "pops-netdisk-icon-click-event-loop-find-sharecode",
                  NetDiskData.iconDefaultClickEventToFindShareCodeByLoop
                )
              ) {
                Qmsg.info("已经定位至最后一个元素了");
                return;
              }
              findGenerator = void 0;
              iterator = void 0;
            }
          }
        );
      },
    },
    /**
     * 显示直链的弹窗
     */
    staticView: {
      getCSS() {
        return `
        .pops-folder-list .list-name-text{
          max-width: 300px;
        }
        .netdisk-static-link-onefile .pops-folder-list .list-name-text{
          max-width: 220px;
        }
        .netdisk-static-link-onefile .pops-mobile-folder-content .pops-folder-list .list-name-text{
          max-width: unset;
        }
        `;
      },
      /**
       * 单文件直链弹窗
       * @param {{
       * title: string,
       * fileName: string,
       * fileType: ?string,
       * fileSize: ?string,
       * downloadUrl: string,
       * fileUploadTime: ?string,
       * fileLatestTime: ?string
       * clickCallBack: ?(_fileDetails_:{
       *    title:string,
       *    fileName:string,
       *    fileType:?string,
       *    fileSize:?string,
       *    downloadUrl:string,
       *    fileUploadTime:?string,
       *    fileLatestTime:?string,
       * })=>{}
       * }} fileDetails 配置
       */
      oneFile(fileDetails) {
        log.success(["成功获取单文件直链", fileDetails]);
        NetDiskPops.folder(
          {
            title: {
              text: fileDetails.title,
            },
            folder: [
              {
                fileName: fileDetails.fileName,
                fileSize: fileDetails.fileSize,
                fileType: fileDetails.fileType ?? "",
                createTime:
                  fileDetails.fileUploadTime || fileDetails.fileLatestTime,
                latestTime:
                  fileDetails.fileLatestTime || fileDetails.fileUploadTime,
                isFolder: false,
                index: 0,
                async clickEvent() {
                  if (typeof fileDetails.clickCallBack === "function") {
                    fileDetails.clickCallBack(fileDetails);
                  } else {
                    return {
                      autoDownload: true,
                      mode: "aBlank",
                      url: fileDetails.downloadUrl,
                    };
                  }
                },
              },
            ],
            btn: {
              ok: {
                text: "下载",
                callback() {
                  if (typeof fileDetails.clickCallBack === "function") {
                    fileDetails.clickCallBack(fileDetails);
                  } else {
                    window.open(fileDetails.downloadUrl, "_blank");
                  }
                },
              },
            },
            class: "netdisk-static-link-onefile",
            style: this.getCSS(),
          },
          NetDiskUI.popsStyle.oneFileStaticView
        );
      },
      /**
       * 多文件直链弹窗
       * @param {string} title 标题
       * @param {?{
       * fileName: string,
       * fileSize: string|number,
       * fileType: ?string,
       * createTime: ?string,
       * latestTime: ?string,
       * isFolder: boolean,
       * index: ?number,
       * clickCallBack: ?(event:Event,_config_: object)=>{}
       * }[]} [folderInfoList=[]] 文件夹信息
       */
      moreFile(title, folderInfoList = []) {
        log.success(["文件解析信息", folderInfoList]);
        NetDiskPops.folder(
          {
            title: {
              text: title,
            },
            folder: folderInfoList,
            style: this.getCSS(),
          },
          NetDiskUI.popsStyle.moreFileStaticView
        );
      },
    },
    /**
     * 需要重新输入新密码的弹窗
     * @param {string} [title="密码错误"] 标题
     * @param {string} [netDiskName=""] 网盘名称
     * @param {number} netDiskIndex 网盘名称索引下标
     * @param {string} shareCode
     * @param {Function} [okCallBack=()=>{}]
     */
    newAccessCodeView(
      title = "密码错误",
      netDiskName = "",
      netDiskIndex,
      shareCode,
      okCallBack = () => {}
    ) {
      const accessCodeConfirm = NetDiskPops.prompt(
        {
          title: {
            text: title,
            position: "center",
            html: false,
          },
          btn: {
            reverse: true,
            position: "end",
            cancel: {
              text: "取消",
            },
            ok: {
              callback: (event) => {
                /* 把输入的新密码去空格 */
                let userInputAccessCode = event.text.replace(/[\s]*/gi, "");
                /* 处理已显示的链接 */
                let uiLink = NetDisk.handleLinkShow(
                  netDiskName,
                  netDiskIndex,
                  shareCode,
                  userInputAccessCode,
                  void 0
                );
                let currentItemSelector = `.netdisk-url a[data-netdisk='${netDiskName}'][data-sharecode='${shareCode}']`;
                let currentHistoryItemSelector = `.netdiskrecord-link a[data-netdisk='${netDiskName}'][data-sharecode='${shareCode}']`;
                let currentItemElement =
                  NetDiskUI.Alias.uiLinkAlias?.$shadowRoot?.querySelector(
                    currentItemSelector
                  );
                let currentHistoryItemElement =
                  NetDiskUI.Alias.historyAlias?.$shadowRoot?.querySelector(
                    currentHistoryItemSelector
                  );
                if (currentItemElement) {
                  currentItemElement.setAttribute(
                    "data-accesscode",
                    userInputAccessCode
                  );
                  DOMUtils.html(currentItemElement, uiLink);
                }
                /* 历史记录的弹出的 */
                if (currentHistoryItemElement) {
                  currentHistoryItemElement.setAttribute(
                    "data-accesscode",
                    userInputAccessCode
                  );
                  DOMUtils.html(currentHistoryItemElement, uiLink);
                }

                log.info(
                  `${netDiskName} 重新输入的密码:${userInputAccessCode}`
                );
                okCallBack(userInputAccessCode);
                event.close();
              },
            },
          },
          content: {
            placeholder: "请重新输入密码",
            focus: true,
          },
        },
        NetDiskUI.popsStyle.inputNewAccessCodeView
      );
      /**
       * 注册键盘回车事件
       */
      utils.listenKeyboard(
        accessCodeConfirm.$shadowRoot,
        "keypress",
        (keyName) => {
          if (keyName === "Enter") {
            accessCodeConfirm.$shadowRoot
              .querySelector(".pops-prompt-btn-ok")
              ?.click();
          }
        }
      );
    },
    /**
     * 网盘历史匹配到的记录弹窗
     */
    netDiskHistoryMatch: {
      /**
       * 本地存储的keyName
       */
      storageKey: "netDiskHistoryMatch",
      /**
       * 是否已设置其它DOM事件
       */
      isSetOtherEvent: false,
      /**
       * 分页
       */
      dataPaging: void 0,
      /**
       * 显示弹窗
       */
      show() {
        let data = this.getNetDiskHistoryMatchData();
        let dataHTML = "";
        let that = this;
        data = this.orderNetDiskHistoryMatchData(data);
        for (let index = 0; index < 10; index++) {
          if (data[index]) {
            dataHTML += that.getTableHTML(data[index]);
          }
        }
        dataHTML = `
        <div class="netdiskrecord-search">
          <input type="text" placeholder="搜索链接/网址/网址标题,可正则搜索">
        </div>
        <div class="netdiskrecord-table"><ul>${dataHTML}</ul></div>
        <div class="netdiskrecord-page">

        </div>`;
        NetDiskUI.Alias.historyAlias = NetDiskPops.confirm(
          {
            title: {
              text: "历史匹配记录",
              position: "center",
            },
            content: {
              text: dataHTML,
              html: true,
            },
            btn: {
              reverse: true,
              position: "space-between",
              ok: {
                enable: true,
                callback(event) {
                  event.close();
                  NetDiskUI.Alias.historyAlias = void 0;
                },
              },
              close: {
                callback(event) {
                  event.close();
                  NetDiskUI.Alias.historyAlias = void 0;
                },
              },
              cancel: {
                enable: false,
              },
              other: {
                enable: true,
                text: `清空所有(${data.length})`,
                type: "xiaomi-primary",
                callback: (event) => {
                  NetDiskPops.confirm({
                    title: {
                      text: "删除",
                      position: "center",
                    },
                    content: {
                      text: "确定清空所有的记录?",
                      html: false,
                    },
                    btn: {
                      ok: {
                        enable: true,
                        callback(okEvent) {
                          that.clearNetDiskHistoryMatchData();
                          DOMUtils.remove(
                            NetDiskUI.Alias.historyAlias.$shadowRoot.querySelectorAll(
                              ".whitesevPopNetDiskHistoryMatch .pops-confirm-content ul li"
                            )
                          );
                          okEvent.close();
                          DOMUtils.html(
                            NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
                              ".whitesevPopNetDiskHistoryMatch .netdiskrecord-page"
                            ),
                            ""
                          );
                          DOMUtils.text(
                            NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
                              ".whitesevPopNetDiskHistoryMatch .pops-confirm-btn-other"
                            ),
                            DOMUtils.text(
                              NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
                                ".whitesevPopNetDiskHistoryMatch .pops-confirm-btn-other"
                              )
                            ).replace(/[\d]+/gi, "0")
                          );
                        },
                      },
                      cancel: {
                        text: "取消",
                        enable: true,
                      },
                    },
                  });
                },
              },
            },
            mask: {
              clickCallBack(originalRun) {
                originalRun();
                NetDiskUI.Alias.historyAlias = null;
              },
            },
            class: "whitesevPopNetDiskHistoryMatch",
            style: this.getCSS(),
          },
          NetDiskUI.popsStyle.historyMatchView
        );
        this.setDataPaging(data);
        this.setEvent(NetDiskUI.Alias.historyAlias.$shadowRoot);
        this.setSearchEvent();
        NetDiskUI.setRightClickMenu(
          NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
            ".whitesevPopNetDiskHistoryMatch"
          ),
          ".netdiskrecord-link a",
          true
        );
      },
      /**
       * 获取CSS
       */
      getCSS() {
        return `
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content ul{

        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content li{
          display: flex;
          flex-direction: column;
          justify-content: center;
          border-radius: 10px;
          box-shadow: 0 0.3px 0.6px rgb(0 0 0 / 6%), 0 0.7px 1.3px rgb(0 0 0 / 8%), 0 1.3px 2.5px rgb(0 0 0 / 10%), 0 2.2px 4.5px rgb(0 0 0 / 12%), 0 4.2px 8.4px rgb(0 0 0 / 14%), 0 10px 20px rgb(0 0 0 / 20%);
          margin: 20px 10px;
          padding: 10px;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-search{
          height: 11%;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-table{
          height: calc( 85% - 40px);
          overflow: auto;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-page{
          display: flex;
          justify-content: center;
          align-items: center;
          margin-top: 10px;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-search input{
          border: none;
          border-bottom: 1px solid #000000;
          padding: 0px 5px;
          line-height: 28px;
          width: -moz-available;
          width: -webkit-fill-available;
          width: fill-available;
          margin: 5px 5px 0px 5px;
          background: none;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-search input:focus-visible{
          outline: none;
          border-bottom: 1px solid #0009ff;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-link{
          
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-link a{
          color: #ff4848;
          font-size: 0.8em;
          border: none;
          word-break: break-word;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-link a[isvisited=true]{
          color: #8b8888;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-icon{
          
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-icon .netdisk-icon-img{
          width: 28px;
          height: 28px;
          min-width: 28px;
          min-height: 28px;
          font-size: 0.8em;
          border-radius: 10px;
          box-shadow: 0 0.3px 0.6px rgb(0 0 0 / 6%), 0 0.7px 1.3px rgb(0 0 0 / 8%), 0 1.3px 2.5px rgb(0 0 0 / 10%), 0 2.2px 4.5px rgb(0 0 0 / 12%), 0 4.2px 8.4px rgb(0 0 0 / 14%), 0 10px 20px rgb(0 0 0 / 20%);
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-url{
          color: #000;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-top-url{
          color: #000;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-functions button.btn-delete{
          background: #263cf3;
          color: #fff;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-functions button.btn-delete:active{
          background: #6e7be8;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-link,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-icon,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-url,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-top-url,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-add-time,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-update-time,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-url-title,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-functions{
          display: flex;
          margin: 5px 0px;
        }
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-link p,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-icon p,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-url p,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-top-url p,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-add-time p,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-update-time p,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-url-title p,
        .whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-functions p{
          min-width: 80px;
          max-width: 80px;
          align-self: center;
        }
        `;
      },
      /**
       * 获取显示出的每一项的html
       * @param {NetDiskHistoryDataOption} data
       * @returns {string}
       */
      getTableHTML(data) {
        let netDiskURL = NetDisk.handleLinkShow(
          data.netDiskName,
          data.netDiskIndex,
          data.shareCode,
          data.accessCode,
          data.matchText
        );
        return `
        <li>
          <div class="netdiskrecord-link">
            <p>链接</p>
            <a  href="javascript:;"
                isvisited="false"
                data-netdisk="${data.netDiskName}"
                data-netdisk-index="${data.netDiskIndex}"
                data-sharecode="${data.shareCode}"
                data-accesscode="${data.accessCode}"
                data-open-enable-key="${data.netDiskName}-open-enable"
                data-static-enable-key="${data.netDiskName}-static-enable"
                data-forward-scheme-key="${
                  data.netDiskName
                }-forward-scheme-enable">${netDiskURL}</a>
          </div>
          <div class="netdiskrecord-icon">
            <p>网盘</p>
            <div class="netdisk-icon-img" style="background: url(${
              NetDiskUI.src.icon[data.netDiskName]
            }) no-repeat;background-size:100%"></div>
          </div>
          ${
            data.url === data.topURL
              ? `
          <div class="netdiskrecord-url">
            <p>网址</p>
            <a href="${data.url}" target="_blank">${data.url}</a>
          </div>
          `
              : `
          <div class="netdiskrecord-url">
            <p>网址</p>
            <a href="${data.url}" target="_blank">${data.url}</a>
          </div>
          <div class="netdiskrecord-top-url">
            <p>TOP网址</p>
            <a href="${data.topURL}" target="_blank">${data.topURL}</a>
          </div>
          `
          }
          <div class="netdiskrecord-url-title">
            <p>网址标题</p>
            ${data.title}
          </div>
          <div class="netdiskrecord-add-time">
            <p>记录时间</p>
            ${utils.formatTime(data.addTime)}
          </div>
          <div class="netdiskrecord-update-time">
            <p>更新时间</p>
            ${utils.formatTime(data.updateTime)}
          </div>
          <div class="netdiskrecord-functions">
            <p>功能</p>
            <button class="btn-delete" data-json='${JSON.stringify(
              data
            )}'>删除</button>
          </div>
        </li>`;
      },
      /**
       * 设置只执行一次的事件
       * @param {HTMLElement} target
       */
      setEvent(target) {
        let that = this;
        NetDiskUI.view.setNetDiskUrlClickEvent(
          target,
          ".whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-link a"
        );

        /**
         * 设置删除按钮点击事件
         */
        DOMUtils.on(
          target,
          "click",
          ".whitesevPopNetDiskHistoryMatch .pops-confirm-content .netdiskrecord-functions button.btn-delete",
          function (event) {
            /* 删除中的遮罩层 */
            let deleteLoading = NetDiskPops.loading({
              parent: target.querySelector(
                ".whitesevPopNetDiskHistoryMatch .pops-confirm-content ul"
              ),
              content: {
                text: "删除中...",
              },
              only: true,
              addIndexCSS: false,
            });
            let clickNode = event.target;
            let dataJSON = clickNode.getAttribute("data-json");
            clickNode.closest("li")?.remove();
            that.deleteNetDiskHistoryMatchData(dataJSON);
            deleteLoading?.close();
            let totalNumberText = DOMUtils.text(
              target.querySelector(
                ".whitesevPopNetDiskHistoryMatch .pops-confirm-btn-other"
              )
            );
            let totalNumber = totalNumberText.match(/[\d]+/gi);
            totalNumber = parseInt(totalNumber[totalNumber.length - 1]);
            totalNumber--;
            totalNumberText = totalNumberText.replace(/[\d]+/gi, totalNumber);
            DOMUtils.text(
              target.querySelector(
                ".whitesevPopNetDiskHistoryMatch .pops-confirm-btn-other"
              ),
              totalNumberText
            );
            let data = that.getNetDiskHistoryMatchData();
            data = that.orderNetDiskHistoryMatchData(data);
            that.dataPaging.refresh(data);
            that.pageChangeCallBack(data, that.dataPaging.CONFIG.currentPage);
          }
        );
      },
      /**
       * @param {object[]} data
       * @param {number} page
       */
      pageChangeCallBack(data, page) {
        let startIndex = (page - 1) * 10;
        let dataHTML = "";
        for (let index = 0; index < 10; index++) {
          if (data[startIndex]) {
            dataHTML += this.getTableHTML(data[startIndex]);
          } else {
            break;
          }
          startIndex++;
        }
        NetDiskUI.Alias.historyAlias.$shadowRoot
          .querySelectorAll(
            ".whitesevPopNetDiskHistoryMatch .netdiskrecord-table ul li"
          )
          .forEach((ele) => ele.remove());
        DOMUtils.append(
          NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
            ".whitesevPopNetDiskHistoryMatch .netdiskrecord-table ul"
          ),
          dataHTML
        );
      },
      /**
       * 设置分页
       * @param {NetDiskHistoryDataOption} data
       */
      setDataPaging(data) {
        let that = this;
        this.dataPaging = new DataPaging({
          data: data,
          pageCount: 10,
          pageStep: pops.isPhone() ? 2 : 4,
          currentPage: 1,
          pageChangeCallBack: function (page) {
            that.pageChangeCallBack(data, page);
          },
        });
        this.dataPaging.addCSS(NetDiskUI.Alias.historyAlias.$shadowRoot);
        this.dataPaging.append(
          NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
            ".whitesevPopNetDiskHistoryMatch .netdiskrecord-page"
          )
        );
      },
      /**
       * 设置搜索框的回车事件
       */
      setSearchEvent() {
        let isSeaching = false; /* 当前搜索的状态 */
        let searchLoading = void 0; /* 搜索中的遮罩层 */
        let that = this;
        function searchEvent() {
          if (isSeaching) {
            return;
          }
          isSeaching = true;
          searchLoading = NetDiskPops.loading({
            parent: NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
              ".whitesevPopNetDiskHistoryMatch .pops-confirm-content ul"
            ),
            content: {
              text: "搜索中...",
            },
            only: true,
            addIndexCSS: false,
          });
          let inputText = NetDiskUI.Alias.historyAlias.$shadowRoot
            .querySelector(
              ".whitesevPopNetDiskHistoryMatch .netdiskrecord-search input"
            )
            .value.trim();
          let data = that.getNetDiskHistoryMatchData();
          data = that.orderNetDiskHistoryMatchData(data);
          if (inputText === "") {
            /* 输入空就关闭遮罩层且恢复style */
            let historyDataHTML = "";
            data.forEach((item, index) => {
              if (index > 9) {
                return;
              }
              historyDataHTML += that.getTableHTML(item);
            });
            NetDiskUI.Alias.historyAlias.$shadowRoot
              .querySelectorAll(
                ".whitesevPopNetDiskHistoryMatch .netdiskrecord-table ul li"
              )
              .forEach((ele) => ele.remove());
            DOMUtils.append(
              NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
                ".whitesevPopNetDiskHistoryMatch .netdiskrecord-table ul"
              ),
              historyDataHTML
            );
            searchLoading?.close();
            isSeaching = false;
            that.setDataPaging(data);
            return;
          }
          let isFind = false; /* 找到的状态 */

          let isFindHTML = ""; /* 找到的链接文本 */
          data.forEach((item) => {
            let netDiskURL = NetDisk.handleLinkShow(
              item.netDiskName,
              item.netDiskIndex,
              item.shareCode,
              item.accessCode,
              item.matchText
            );
            if (
              netDiskURL.match(new RegExp(inputText, "ig")) ||
              item.url.match(new RegExp(inputText, "ig")) ||
              item.topURL.match(new RegExp(inputText, "ig")) ||
              item.title.match(new RegExp(inputText, "ig"))
            ) {
              /* 匹配到 */
              isFind = true;
              isFindHTML += that.getTableHTML(item);
            }
          });
          DOMUtils.remove(
            NetDiskUI.Alias.historyAlias.$shadowRoot.querySelectorAll(
              ".whitesevPopNetDiskHistoryMatch .netdiskrecord-table ul li"
            )
          );
          DOMUtils.append(
            NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
              ".whitesevPopNetDiskHistoryMatch .netdiskrecord-table ul"
            ),
            isFindHTML
          );
          DOMUtils.remove(
            NetDiskUI.Alias.historyAlias.$shadowRoot.querySelectorAll(
              ".whitesevPopNetDiskHistoryMatch .netdiskrecord-page > *"
            )
          );
          searchLoading?.close();
          searchLoading = void 0;
          isSeaching = false;
        }

        utils.listenKeyboard(
          NetDiskUI.Alias.historyAlias.$shadowRoot.querySelector(
            ".whitesevPopNetDiskHistoryMatch .netdiskrecord-search input"
          ),
          "keypress",
          (keyName) => {
            if (keyName === "Enter") {
              searchEvent();
            }
          }
        );
      },
      /**
       * 排序数据
       * @returns {NetDiskHistoryDataOption[]}
       */
      orderNetDiskHistoryMatchData(data) {
        let localOrder = GM_getValue(
          "netdisk-history-match-ordering-rule",
          "按 更新时间 - 降序"
        );
        let isDesc =
          localOrder.indexOf("降序") !== -1 ? true : false; /* 降序 */
        let orderField =
          localOrder.indexOf("记录时间") !== -1
            ? "addTime"
            : "updateTime"; /* 排序字段 */

        utils.sortListByProperty(
          data,
          (item) => {
            return item[orderField];
          },
          isDesc
        );
        return data;
      },
      /**
       * 存储匹配到的链接
       * @param {string} netDiskName 网盘名称
       * @param {number} netDiskIndex 网盘名称索引下标
       * @param {string} shareCode 分享码
       * @param {string} accessCode 访问码
       * @param {string} matchText 匹配到的文本
       * @returns
       */
      setNetDiskHistoryMatchData(
        netDiskName,
        netDiskIndex,
        shareCode,
        accessCode,
        matchText
      ) {
        if (!GM_getValue("saveMatchNetDisk", false)) {
          return;
        }
        let localData = this.getNetDiskHistoryMatchData();
        for (let index = 0; index < localData.length; index++) {
          const data = localData[index];
          if (
            data.url === window.location.href &&
            data.topURL === top.window.location.href &&
            data.netDiskName === netDiskName &&
            shareCode.startsWith(data.shareCode) &&
            data.netDiskIndex === netDiskIndex
          ) {
            if (data.matchText !== matchText) {
              /* 修改/设置新的matchText */
              log.success(["匹配历史记录 -> 设置新的matchText", [matchText]]);
              localData[index].matchText = matchText;
              localData[index].updateTime = Date.now();
              if (localData[index].title) {
                localData[index].title = document.title;
              }
              GM_setValue(this.storageKey, localData);
              return;
            }
            if (data.accessCode !== accessCode) {
              /* 修改accessCode */
              log.success("匹配历史记录 -> 修改accessCode");
              localData[index].accessCode = accessCode;
              localData[index].updateTime = Date.now();
              if (localData[index].title) {
                localData[index].title = document.title;
              }
              GM_setValue(this.storageKey, localData);
              return;
            }
            if (data.accessCode === accessCode) {
              /* 已存在一模一样的 */
              return;
            }
          }
        }
        log.success("匹配历史记录 -> 增加新的");
        let time = Date.now();
        localData = [
          ...localData,
          {
            url: window.location.href,
            topURL: top.window.location.href,
            netDiskName: netDiskName,
            netDiskIndex: netDiskIndex,
            shareCode: shareCode,
            accessCode,
            addTime: time,
            updateTime: time,
            title: document.title || top.document.title,
            matchText: matchText,
          },
        ];
        GM_setValue(this.storageKey, localData);
      },
      /**
       * 获取历史匹配到的链接
       * @returns {NetDiskHistoryDataOption[]}
       */
      getNetDiskHistoryMatchData() {
        let data = GM_getValue(this.storageKey);
        if (data == void 0) {
          data = [];
          GM_setValue(this.storageKey, data);
        }
        return data;
      },
      /**
       * 检测并尝试修复本地的数据
       */
      checkAndRepairLocalData() {
        let repairCount = 0;
        let data = GM_getValue(this.storageKey);
        if (Array.isArray(data)) {
          for (let index = 0; index < data.length; index++) {
            const itemData = data[index];
            if (typeof itemData["matchText"] !== "string") {
              itemData["matchText"] = "";
              repairCount++;
            }
          }
        } else {
          data = [];
        }
        GM_setValue(this.storageKey, data);
        return {
          count: data.length,
          repairCount: repairCount,
        };
      },
      /**
       * 删除存储的某个项
       * @param {string} dataJSONText
       */
      deleteNetDiskHistoryMatchData(dataJSONText) {
        let isSuccess = false;
        let data = this.getNetDiskHistoryMatchData();
        for (let index = 0; index < data.length; index++) {
          if (JSON.stringify(data[index]) === dataJSONText) {
            log.success("删除 ===> ", data[index]);
            data.splice(index, 1);
            isSuccess = true;
            break;
          }
        }
        if (isSuccess) {
          GM_setValue(this.storageKey, data);
        }
        return isSuccess;
      },
      /**
       * 清空所有配置
       */
      clearNetDiskHistoryMatchData() {
        GM_setValue(this.storageKey, []);
      },
    },
    /**
     * 自定义访问码规则,用于设置某个网站下的某个网盘链接的固定访问码
     */
    accessCodeRule: {
      /**
       * 弹窗的className
       */
      accessCodeRuleDialogClassName: "whitesevPopNetDiskAccessCodeRule",
      /**
       * 显示弹窗
       */
      show() {
        let that = this;
        let popsConfirm = NetDiskPops.confirm(
          {
            title: {
              text: "自定义访问码规则",
              position: "center",
            },
            content: {
              text: `
            <div class="netdisk-accesscode-rule-table">
              <ul>
              ${that.getShowItemHTML()}
              </ul>
            </div>
            `,
              html: true,
            },
            btn: {
              merge: true,
              reverse: false,
              position: "space-between",
              ok: {
                enable: true,
                text: "添加",
                callback(event) {
                  that.showRule(event);
                },
              },
              close: {
                callback(event) {
                  event.close();
                },
              },
              cancel: {
                enable: true,
              },
              other: {
                enable: true,
                type: "xiaomi-primary",
                text: `清空所有`,
                callback(event) {
                  NetDiskPops.confirm({
                    title: {
                      text: "删除",
                      position: "center",
                    },
                    content: {
                      text: "确定清空所有的规则?",
                      html: false,
                    },
                    btn: {
                      ok: {
                        enable: true,
                        callback: function (okEvent) {
                          log.success("清空所有");
                          that.deleteAllValue();
                          if (that.getValue().length) {
                            Qmsg.error("清空全部规则失败");
                            return;
                          } else {
                            Qmsg.success("已清空全部规则");
                          }
                          that.setDeleteAllBtnText(event.animElement);
                          event.animElement.querySelector(
                            ".pops-confirm-content ul"
                          ).innerHTML = "";
                          okEvent.close();
                        },
                      },
                      cancel: {
                        text: "取消",
                        enable: true,
                      },
                    },
                  });
                },
              },
            },
            class: this.accessCodeRuleDialogClassName,
            style: this.getCSS(),
          },
          NetDiskUI.popsStyle.accessCodeRuleView
        );
        that.setDeleteAllBtnText(popsConfirm.animElement);
        this.setEvent(popsConfirm);
      },
      getShowItemHTML() {
        let result = "";
        this.getValue().forEach((item) => {
          let netdiskName = "";
          item.netdisk.forEach((_netdisk_) => {
            netdiskName += _netdisk_.name;
            netdiskName += "、";
          });
          netdiskName = netdiskName.replace(/、$/g, "");
          result += `
          <li>
            <div class="accesscode-rule-url-regexp">
              <p>匹配规则</p>
              ${item.urlRegexp}
            </div>
            <div class="accesscode-rule-netdisk-name">
              <p>匹配网盘</p>
              ${netdiskName}
            </div>
            <div class="accesscode-rule-accesscode">
              <p>固定值</p>
              ${item.accessCode}
            </div>
            <div class="accesscode-rule-functions" data-json='${JSON.stringify(
              item
            )}'>
              <p>功能</p>
              <button style="background: #46cb31;color: #fff;" data-edit>修改</button>
              <button style="background: #263cf3;color: #fff;" data-delete>删除</button>
            </div>
          </li>
          `;
        });
        return result;
      },
      getCSS() {
        return `
        .pops-confirm-content .whitesev-accesscode-rule{
          display: flex;
          align-items: center;
          justify-content: space-between;
          margin: 15px 15px;
        }
        
        .pops-confirm-content div.netdisk-accesscode-rule-table{
          /* height: calc( 85% - 40px); */
          overflow: auto;
        }

        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-url-regexp,
        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-netdisk-name,
        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-accesscode,
        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-functions{
          display: flex;
          margin: 5px 0px;
        }

        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-url-regexp p,
        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-netdisk-name p,
        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-accesscode p,
        .pops-confirm-content .netdisk-accesscode-rule-table .accesscode-rule-functions p{
          min-width: 80px;
          max-width: 80px;
          align-self: center;
        }
        .pops-confirm-content .netdisk-accesscode-rule-table li {
          display: flex;
          flex-direction: column;
          justify-content: center;
          border-radius: 10px;
          box-shadow: 0 0.3px 0.6px rgb(0 0 0 / 6%), 0 0.7px 1.3px rgb(0 0 0 / 8%), 0 1.3px 2.5px rgb(0 0 0 / 10%), 0 2.2px 4.5px rgb(0 0 0 / 12%), 0 4.2px 8.4px rgb(0 0 0 / 14%), 0 10px 20px rgb(0 0 0 / 20%);
          margin: 20px 10px;
          padding: 10px;
        }
        `;
      },
      /**
       * 显示规则弹窗进行添加/修改
       * @param {object} mainEvent
       * @param {boolean} isEdit 是否是修改模式
       * @param {{
       * urlRegexp: string,
       * netdisk: {
       *  name:string,
       *  value:string,
       * }[],
       * accessCode: string,
       * }} oldValue 当isEdie为true时传入的值
       */
      showRule(mainEvent, isEdit = false, oldValue) {
        let that = this;
        let popsConfirm = NetDiskPops.confirm(
          {
            title: {
              text: isEdit ? "修改规则" : "添加规则",
              position: "center",
            },
            content: {
              text: `
              <div class="whitesev-accesscode-rule">
                <div type-name>匹配网站</div>
                <div class="pops-panel-input">
                  <input type="text" placeholder="请输入需要匹配的正则规则" val-key="access-rule-url" />
                </div>
              </div>
              <div class="whitesev-accesscode-rule">
                <div>匹配网盘</div>
                <div class="pops-panel-select">
                  <select val-key="access-rule-netdisk" multiple="true" style="height: auto;">
                    <option data-value="baidu">百度网盘</option>
                    <option data-value="lanzou">蓝奏云</option>
                    <option data-value="tianyiyun">天翼云</option>
                    <option data-value="hecaiyun">中国移动云盘</option>
                    <option data-value="aliyun">阿里云</option>
                    <option data-value="wenshushu">文叔叔</option>
                    <option data-value="nainiu">奶牛</option>
                    <option data-value="_123pan">123盘</option>
                    <option data-value="weiyun">微云</option>
                    <option data-value="xunlei">迅雷云盘</option>
                    <option data-value="_115pan">115网盘</option>
                    <option data-value="chengtong">城通网盘</option>
                    <option data-value="kuake">夸克网盘</option>
                    <option data-value="jianguoyun">坚果云</option>
                    <option data-value="onedrive">OneDrive</option>
                  </select>
                </div>
              </div>
              <div class="whitesev-accesscode-rule">
                <div>固定值</div>
                <div class="pops-panel-input">
                  <input type="text" placeholder="请输入固定的访问码" val-key="access-rule-accesscode" />
                </div>
              </div>
            `,
              html: true,
            },
            btn: {
              ok: {
                enable: true,
                text: isEdit ? "修改" : "添加",
                callback(event) {
                  let accessRuleUrl = event.popsElement.querySelector(
                    'input[val-key="access-rule-url"]'
                  ).value;
                  /**
                   * @type {{name:string,value:string}[]}
                   */
                  let accessRuleNetDisk = [];
                  let accessRuleNetDiskElement =
                    event.popsElement.querySelector(
                      'select[val-key="access-rule-netdisk"]'
                    );
                  Array.from(accessRuleNetDiskElement.selectedOptions).forEach(
                    (item) => {
                      accessRuleNetDisk.push({
                        name: item.value,
                        value: item.getAttribute("data-value"),
                      });
                    }
                  );
                  let accessRuleAccessCode = event.popsElement.querySelector(
                    'input[val-key="access-rule-accesscode"]'
                  ).value;
                  if (!that.checkRuleUrlValid(accessRuleUrl)) {
                    log.error(["验证失败", accessRuleUrl]);
                    return;
                  }
                  if (isEdit) {
                    /* 编辑 */
                    if (
                      that.changeValue(oldValue, {
                        urlRegexp: accessRuleUrl,
                        netdisk: accessRuleNetDisk,
                        accessCode: accessRuleAccessCode,
                      })
                    ) {
                      log.success([
                        "修改成功",
                        {
                          urlRegexp: accessRuleUrl,
                          netdisk: accessRuleNetDisk,
                          accessCode: accessRuleAccessCode,
                        },
                      ]);
                      Qmsg.success("修改成功");
                      mainEvent.animElement.querySelector(
                        ".netdisk-accesscode-rule-table ul"
                      ).innerHTML = that.getShowItemHTML();
                      popsConfirm.close();
                    } else {
                      Qmsg.error("修改失败");
                    }
                  } else {
                    /* 添加 */
                    if (
                      that.addValue({
                        urlRegexp: accessRuleUrl,
                        netdisk: accessRuleNetDisk,
                        accessCode: accessRuleAccessCode,
                      })
                    ) {
                      Qmsg.success("添加成功");
                      mainEvent.animElement.querySelector(
                        ".netdisk-accesscode-rule-table ul"
                      ).innerHTML = that.getShowItemHTML();
                      that.setDeleteAllBtnText(mainEvent.animElement);
                      popsConfirm.close();
                    } else {
                      Qmsg.error("已存在重复的规则");
                    }
                  }
                },
              },
              cancel: {
                text: "取消",
                enable: true,
              },
            },
            class: "whitesevPopNetDiskAccessCodeRuleAddOrEdit",
            style: `
            ${pops.config.cssText.panelCSS}

            .whitesevPopNetDiskAccessCodeRuleAddOrEdit .whitesev-accesscode-rule{
              display: flex;
              justify-content: space-between;
              margin: 4px 4px;
            }
            .whitesevPopNetDiskAccessCodeRuleAddOrEdit .whitesev-accesscode-rule select{
              height: 150px;
            }
            `,
          },
          NetDiskUI.popsStyle.accessCodeRuleEditView
        );
        this.setRuleEvent(popsConfirm.element);
        if (isEdit) {
          popsConfirm.element.querySelector(
            '.whitesev-accesscode-rule input[val-key="access-rule-url"]'
          ).value = oldValue.urlRegexp;
          let optionElement = popsConfirm.element.querySelectorAll(
            '.whitesev-accesscode-rule select[val-key="access-rule-netdisk"] option'
          );
          oldValue.netdisk.forEach((item) => {
            optionElement.forEach((element) => {
              if (element.getAttribute("data-value") === item.value) {
                element.selected = true;
                log.success(["选中", element]);
                return;
              }
            });
          });
          popsConfirm.element.querySelector(
            '.whitesev-accesscode-rule input[val-key="access-rule-accesscode"]'
          ).value = oldValue.accessCode;
        }
      },
      /**
       * 修改 删除所有(xx)的文字
       * @param {HTMLElement} element
       */
      setDeleteAllBtnText(element) {
        if (
          element.querySelector(
            ".pops-confirm-btn button.pops-confirm-btn-other"
          )
        ) {
          element.querySelector(
            ".pops-confirm-btn button.pops-confirm-btn-other"
          ).textContent = `清空所有(${this.getValue().length})`;
        } else if (element.getRootNode() instanceof ShadowRoot) {
          element
            .getRootNode()
            .querySelector(
              ".whitesevPopNetDiskAccessCodeRule .pops-confirm-btn button.pops-confirm-btn-other"
            ).textContent = `清空所有(${this.getValue().length})`;
        }
      },
      /**
       * 校验填写的匹配网站正则规则是否正确
       * @param {string} accessRuleUrl 填写的匹配网站正则规则
       * @returns {boolean}
       */
      checkRuleUrlValid(accessRuleUrl) {
        if (utils.isNull(accessRuleUrl)) {
          Qmsg.error("匹配网站的正则不能为空或纯空格");
          return false;
        }
        try {
          new RegExp(accessRuleUrl);
        } catch (error) {
          log.error(error);
          Qmsg.error("匹配网站的正则错误</br>" + error.message, {
            html: true,
            timeout: 5000,
          });
          return false;
        }
        return true;
      },
      /**
       * 设置事件
       * @param {object} event
       */
      setEvent(event) {
        let that = this;
        DOMUtils.on(
          event.animElement,
          "click",
          ".netdisk-accesscode-rule-table div.accesscode-rule-functions button[data-delete]",
          function () {
            let dataJSON = this.closest(
              ".accesscode-rule-functions"
            ).getAttribute("data-json");
            dataJSON = utils.toJSON(dataJSON);
            log.success(["删除👉", dataJSON]);
            if (that.deleteValue(dataJSON)) {
              this.closest("li").remove();
              that.setDeleteAllBtnText(event.animElement);
            } else {
              Qmsg.error("删除失败");
            }
          }
        );
        DOMUtils.on(
          event.element,
          "click",
          ".netdisk-accesscode-rule-table div.accesscode-rule-functions button[data-edit]",
          function () {
            let dataJSON = this.closest(
              ".accesscode-rule-functions"
            ).getAttribute("data-json");
            dataJSON = utils.toJSON(dataJSON);
            log.success(["修改👉", dataJSON]);
            let newEvent = Object.assign({}, event);
            newEvent.animElement = newEvent.element;
            that.showRule(newEvent, true, dataJSON);
          }
        );
      },
      /**
       * 设置事件
       * @param {HTMLElement} element 弹窗元素
       */
      setRuleEvent(element) {},
      /**
       * 获取值
       * @returns {{
       * urlRegexp: string,
       * netdisk: {name:string,value:string}[]
       * accessCode: string,
       * }[]}
       */
      getValue() {
        return GM_getValue("accessCodeRule", []);
      },
      /**
       * 设置值
       * @param {{
       * urlRegexp: string,
       * netdisk: {name:string,value:string}[]
       * accessCode: string,
       * }} value
       */
      setValue(value) {
        let localData = this.getValue();
        localData.push(value);
        GM_setValue("accessCodeRule", localData);
      },
      /**
       * 修改值
       * @param {{
       * urlRegexp: string,
       * netdisk: {name:string,value:string}[]
       * accessCode: string,
       * }} oldValue
       * @param {{
       * urlRegexp: string,
       * netdisk: {name:string,value:string}[]
       * accessCode: string,
       * }} newValue
       * @returns {boolean} 是否修改成功
       */
      changeValue(oldValue, newValue) {
        let result = false;
        let localData = this.getValue();
        let oldValueStr = JSON.stringify(oldValue);
        for (let i = 0; i < localData.length; i++) {
          if (JSON.stringify(localData[i]) === oldValueStr) {
            localData[i] = newValue;
            result = true;
            break;
          }
        }
        GM_setValue("accessCodeRule", localData);
        return result;
      },
      /**
       * 添加值
       * @param {{
       * urlRegexp: string,
       * netdisk: {name:string,value:string}[]
       * accessCode: string,
       * }} value
       */
      addValue(value) {
        let result = true;
        let localData = this.getValue();
        for (let i = 0; i < localData.length; i++) {
          if (
            localData[i].urlRegexp === value.urlRegexp &&
            localData[i].netdisk === value.netdisk
          ) {
            result = false;
            break;
          }
        }
        if (result) {
          localData.push(value);
          this.setValue(value);
        }
        return result;
      },
      /**
       * 删除值
       */
      deleteValue(value) {
        let result = false;
        let localData = this.getValue();
        let valueStr = JSON.stringify(value);
        for (let i = 0; i < localData.length; i++) {
          if (JSON.stringify(localData[i]) === valueStr) {
            localData.splice(i, 1);
            result = true;
            break;
          }
        }
        if (result) {
          GM_setValue("accessCodeRule", localData);
        }
        return result;
      },
      /**
       * 清空所有规则
       */
      deleteAllValue() {
        GM_setValue("accessCodeRule", []);
      },
    },
    /**
     * 主动识别文本
     */
    matchPasteText: {
      show() {
        let popsConfirm = NetDiskPops.confirm(
          {
            title: {
              text: "主动识别文本",
              position: "center",
            },
            content: {
              text: `
            <textarea class="netdisk-match-paste-text"></textarea>
            `,
              html: true,
            },
            btn: {
              ok: {
                text: "识别",
                callback() {
                  let inputText =
                    popsConfirm.popsElement?.querySelector(
                      ".netdisk-match-paste-text"
                    )?.value || "";
                  if (inputText.trim() !== "") {
                    /* 排除中文 */
                    inputText = inputText.replace(/[\u4e00-\u9fa5]/g, "");
                    NetDiskWorker.postMessage({
                      textList: [inputText],
                      matchTextRange: GM_getValue(
                        "pageMatchRange",
                        "all"
                      ).toLowerCase(),
                      regular: NetDisk.regular,
                      startTime: Date.now(),
                      from: "PasteText",
                    });
                  }
                },
              },
            },
            class: "whitesevPopNetDiskMatchPasteText",
            style: this.getCSS(),
          },
          NetDiskUI.popsStyle.matchPasteTextView
        );
        popsConfirm.popsElement.querySelector("textarea").focus();
      },
      getCSS() {
        return `
        .pops[type-value=confirm] .pops-confirm-content{
          overflow: hidden;
        }
        .netdisk-match-paste-text {
          --textarea-bd-color: #dcdfe6;
          display: inline-block;
          resize: vertical;
          padding: 5px 15px;
          line-height: 1.5;
          box-sizing: border-box;
          color: #606266;
          border: 1px solid var(--textarea-bd-color);
          border-radius: 4px;
          transition: border-color 0.2s cubic-bezier(0.645, 0.045, 0.355, 1);
          outline: none;
          margin: 0;
          -webkit-appearance: none;
          -moz-appearance: none;
          appearance: none;
          background: none;
          width: 100%;
          height: 100%;
          appearance: none;
          resize: none;
        }
        .netdisk-match-paste-text:hover{
          --textarea-bd-color: #c0c4cc;
        }
        .netdisk-match-paste-text:focus{
          --textarea-bd-color: #3677f0;
        }
        `;
      },
      /**
       *
       * @param {NetDiskWorkerCallBackOptions} data
       */
      matchEndCallBack(data) {
        if (data.data.length) {
          Qmsg.success(
            `成功匹配${data.data.length}个,用时${
              Date.now() - data.startTime
            }ms`
          );
        } else {
          Qmsg.error("未识别到链接");
        }
      },
    },
    /**
     * 监听页面节点内容或节点文本的变动,从而进行匹配网盘链接
     */
    monitorDOMInsert() {
      NetDiskWorker.initWorkerBlobLink();
      NetDiskWorker.initWorker();
      /**
       * 设置-判定为添加元素才进行匹配
       * @type {boolean}
       */
      const isAddedNodesToMatch = GM_getValue("isAddedNodesToMatch");
      /**
       * 观察者的事件
       * @param {?MutationRecord[]} mutations
       * @returns
       */
      async function observeEvent(mutations) {
        if (NetDiskUI.isHandleMatch) {
          /* 当前正在处理文本正则匹配中 */
          return;
        }
        if (isAddedNodesToMatch && mutations && mutations.length) {
          let hasAddNodes = false;
          for (const mutation of mutations) {
            if (
              mutation.addedNodes &&
              mutation.addedNodes instanceof NodeList
            ) {
              if (mutation.addedNodes.length) {
                hasAddNodes = true;
                break;
              }
            }
          }
          if (!hasAddNodes) {
            return;
          }
        }
        NetDiskUI.isHandleMatch = true;
        /** 开始时间 */
        const startTime = Date.now();
        if (GM_getValue("readClipboard", false)) {
          NetDisk.clipboardText = await NetDisk.getClipboardText();
        }
        if (typeof NetDisk.clipboardText !== "string") {
          NetDisk.clipboardText = "";
        }
        let matchTextRangeLowerCase = GM_getValue(
          "pageMatchRange",
          "all"
        ).toLowerCase();
        let matchTextList = [];

        /* 剪贴板内容 */
        if (NetDisk.clipboardText.trim() !== "") {
          matchTextList.push(NetDisk.clipboardText);
        }
        /* 当前的网页链接 */
        if (NetDisk.allowMatchLocationHref) {
          let decodeUrl = window.location.href;
          try {
            decodeUrl = decodeURIComponent(decodeUrl);
          } catch (error) {}
          matchTextList.push(decodeUrl);
        }
        /* innerText和innerHTML */
        if (matchTextRangeLowerCase === "all") {
          let pageText = NetDisk.ignoreStrRemove(
            document.documentElement.innerText
          );
          let pageHTML = NetDisk.ignoreStrRemove(
            document.documentElement.innerHTML,
            true
          );
          matchTextList.push(pageText);
          matchTextList.push(pageHTML);
        } else if (matchTextRangeLowerCase === "innertext") {
          /* innerText */
          let pageText = NetDisk.ignoreStrRemove(
            document.documentElement.innerText
          );
          matchTextList.push(pageText);
        } else {
          /* innerHTML */
          let pageHTML = NetDisk.ignoreStrRemove(
            document.documentElement.innerHTML,
            true
          );
          matchTextList.push(pageHTML);
        }
        /* 发送消息 */
        NetDiskWorker.postMessage({
          textList: matchTextList,
          matchTextRange: matchTextRangeLowerCase,
          regular: NetDisk.regular,
          startTime: startTime,
          from: "DOMChange",
        });
      }
      utils.mutationObserver(document.documentElement, {
        callback: observeEvent,
        config: {
          /* 子节点的变动(新增、删除或者更改) */
          childList: true,
          /* 节点内容或节点文本的变动 */
          characterData: true,
          /* 是否将观察器应用于该节点的所有后代节点 */
          subtree: true,
        },
      });
      /* 主动触发一次,因为有的页面上没触发mutationObserver */
      observeEvent();
    },
    /**
     * 设置标题的右键菜单
     */
    setGlobalRightClickMenu(element) {
      NetDiskUI.view.registerContextMenu(element, void 0, [
        {
          text: "设置",
          callback() {
            log.info("打开-设置");
            NetDiskUI.suspension.showSettingView();
          },
        },
        {
          text: "历史匹配记录",
          callback() {
            log.info("打开-历史匹配记录");
            NetDiskUI.netDiskHistoryMatch.show();
          },
        },
        {
          text: "访问码规则",
          callback() {
            log.info("打开-访问码规则");
            NetDiskUI.accessCodeRule.show();
          },
        },
        {
          text: "自定义规则",
          callback() {
            log.info("打开-自定义规则");
            NetDiskCustomRules.showUI(false);
          },
        },
        {
          text: "主动识别文本",
          callback() {
            log.info("打开-主动识别文本");
            NetDiskUI.matchPasteText.show();
          },
        },
      ]);
    },
    /**
     * 设置右键菜单
     * @param {HTMLElement | Window} target
     * @param {string} selector
     * @param {boolean} isHistoryView
     */
    setRightClickMenu(target, selector, isHistoryView) {
      let showTextList = [
        {
          text: "复制链接",
          callback: function (event, contextMenuEvent) {
            let linkElement = contextMenuEvent.target;
            const { netDiskName, netDiskIndex, shareCode, accessCode } =
              NetDiskUI.getElementNetDiskLinkData(linkElement);
            NetDiskParse.copyText(
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
          },
        },
        {
          text: "访问链接",
          callback: function (event, contextMenuEvent) {
            let linkElement = contextMenuEvent.target;
            const { netDiskName, netDiskIndex, shareCode, accessCode } =
              NetDiskUI.getElementNetDiskLinkData(linkElement);
            let url = NetDiskParse.getBlankUrl(
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
            NetDiskParse.blank(
              url,
              netDiskName,
              netDiskIndex,
              shareCode,
              accessCode
            );
          },
        },
        {
          text: "修改访问码",
          callback: function (event, contextMenuEvent) {
            let linkElement = contextMenuEvent.target;
            const { netDiskName, netDiskIndex, shareCode, accessCode } =
              NetDiskUI.getElementNetDiskLinkData(linkElement);
            function newAccessCodeByHistoryViewCallBack(userInputAccessCode) {
              let currentTime = new Date().getTime();
              let data = GM_getValue(NetDiskUI.netDiskHistoryMatch.storageKey);
              let editFlag = false;
              data.forEach((item) => {
                if (
                  item["netDiskName"] === netDiskName &&
                  item["netDiskIndex"] === netDiskIndex &&
                  item["shareCode"] === shareCode &&
                  item["accessCode"] === accessCode
                ) {
                  item = utils.assign(item, {
                    accessCode: userInputAccessCode,
                    updateTime: currentTime,
                  });
                  log.success(["成功找到项", item]);
                  editFlag = true;
                  return;
                }
              });
              if (editFlag) {
                GM_setValue(NetDiskUI.netDiskHistoryMatch.storageKey, data);
                linkElement
                  .closest("li")
                  .querySelector(
                    ".netdiskrecord-update-time"
                  ).lastChild.textContent = utils.formatTime(currentTime);
                linkElement.setAttribute(
                  "data-accesscode",
                  userInputAccessCode
                );
                Qmsg.success(
                  `
                  <div style="text-align: left;">旧: ${accessCode}</div>
                  <div style="text-align: left;">新: ${userInputAccessCode}</div>`,
                  {
                    html: true,
                  }
                );
              } else {
                Qmsg.error("修改失败");
              }
            }
            function newAccessCodeCallBack_(userInputAccessCode) {
              event.target.setAttribute("data-accesscode", userInputAccessCode);
              let netDiskDict = NetDisk.linkDict.get(netDiskName);
              if (netDiskDict.has(shareCode)) {
                let currentDict = netDiskDict.get(shareCode);
                netDiskDict.set(
                  shareCode,
                  NetDisk.getLinkDickObj(
                    userInputAccessCode,
                    netDiskIndex,
                    true,
                    currentDict.matchText
                  )
                );
                Qmsg.success(
                  `
                  <div style="text-align: left;">旧: ${accessCode}</div>
                  <div style="text-align: left;">新: ${userInputAccessCode}</div>`,
                  {
                    html: true,
                  }
                );
              } else {
                Qmsg.error("该访问码不在已获取的字典中:" + shareCode);
              }
            }
            NetDiskUI.newAccessCodeView(
              this.text,
              netDiskName,
              netDiskIndex,
              shareCode,
              (userInputAccessCode) => {
                if (isHistoryView) {
                  newAccessCodeByHistoryViewCallBack(userInputAccessCode);
                } else {
                  newAccessCodeCallBack_(userInputAccessCode);
                }
              }
            );
          },
        },
      ];
      NetDiskUI.view.registerContextMenu(target, selector, showTextList);
    },
    /**
     * 获取元素.netdisk-link上的data属性
     * @param {HTMLAnchorElement} $netDiskLink
     */
    getElementNetDiskLinkData($netDiskLink) {
      let netDiskName = $netDiskLink.getAttribute("data-netdisk");
      let netDiskIndex = parseInt(
        $netDiskLink.getAttribute("data-netdisk-index")
      );
      if (isNaN(netDiskIndex)) {
        log.warn(["元素上的netDiskIndex的值是NaN", $netDiskLink]);
        netDiskIndex = 0;
      }
      let shareCode = $netDiskLink.getAttribute("data-sharecode");
      let accessCode = $netDiskLink.getAttribute("data-accesscode");
      return {
        netDiskName,
        netDiskIndex,
        shareCode,
        accessCode,
      };
    },
  };

  /** 全局快捷键 */
  const NetDiskShortcut = {
    /**
     * 是否正在等待用户按下键盘
     */
    isWaitUserPressKeyboard: false,
    /**
     * 获取本地存储的值
     * @param {?string} key
     * @param {?string} defaultVal
     * @returns {{
     * key: string,
     * value: {
     * keyName: string,
     * keyValue: number|string,
     * ohterCodeList: string[]
     * }
     * }|{
     * keyName: string,
     * keyValue: number|string,
     * ohterCodeList: string[]
     * }[]}
     */
    getValue(key, defaultVal) {
      let value = GM_getValue("GM_shortcut", []);
      if (key) {
        for (let index = 0; index < value.length; index++) {
          let item = value[index];
          if (item["key"] === key) {
            return item["value"];
          }
        }
        return defaultVal;
      } else {
        return value;
      }
    },
    /**
     * 删除本地存储的快捷键值
     * @param {string} key
     */
    deleteValue(key) {
      let value = this.getValue();
      let findValueIndex = value.findIndex((item) => item["key"] === key);
      if (findValueIndex !== -1) {
        value.splice(findValueIndex, 1);
      }
      GM_setValue("GM_shortcut", value);
    },
    /**
     * 保存设置的快捷键到本地存储
     * @param {string} key
     * @param {string} keyName
     * @param {string} keyValue
     * @param {string[]} ohterCodeList
     */
    setValue(key, keyName, keyValue, ohterCodeList) {
      let value = this.getValue();
      value.push({
        key: key,
        value: {
          keyName: keyName,
          keyValue: keyValue,
          ohterCodeList: ohterCodeList,
        },
      });
      GM_setValue("GM_shortcut", value);
    },
    /**
     * 获取快捷键显示的文字
     * @param {string} key
     * @param {string} defaultValue
     */
    getShowText(key, defaultValue) {
      let localValue = this.getValue(key);
      if (localValue) {
        /* 如果获取到,转需要显示的文字 */
        let result = "";
        localValue["ohterCodeList"].forEach((item) => {
          result += utils.stringTitleToUpperCase(item, true) + " + ";
        });
        result += localValue["keyName"];
        return result;
      } else {
        /* 未获取到,显示为默认的文字 */
        return defaultValue;
      }
    },
    /**
     * 快捷键按钮录入的点击事件
     */
    buttonClickCallBack(event, key, defaultValue) {
      let localValue = this.getValue(key, defaultValue);
      let spanElement = event.target
        .closest(".pops-panel-button")
        .querySelector("span");
      if (localValue === defaultValue) {
        /* 设置快捷键 */
        let loadingQmsg = Qmsg.loading("请按下快捷键...", {
          showClose: true,
          onClose() {
            keyboardListener.removeListen();
          },
        });
        NetDiskShortcut.isWaitUserPressKeyboard = true;
        let keyboardListener = utils.listenKeyboard(
          window,
          "keyup",
          (keyName, keyValue, ohterCodeList) => {
            let shortcutJSONString = JSON.stringify({
              keyName: keyName,
              keyValue: keyValue,
              ohterCodeList: ohterCodeList,
            });
            let allDetails = this.getValue();
            for (let index = 0; index < allDetails.length; index++) {
              if (
                shortcutJSONString ===
                JSON.stringify(allDetails[index]["value"])
              ) {
                Qmsg.error(
                  `快捷键 ${this.getShowText(
                    allDetails[index]["key"]
                  )} 已被占用`
                );
                NetDiskShortcut.isWaitUserPressKeyboard = false;
                loadingQmsg.close();
                return;
              }
            }
            this.setValue(key, keyName, keyValue, ohterCodeList);
            spanElement.innerHTML = this.getShowText(key, defaultValue);
            NetDiskShortcut.isWaitUserPressKeyboard = false;
            loadingQmsg.close();
          }
        );
      } else {
        /* 清空快捷键 */
        this.deleteValue(key);
      }

      spanElement.innerHTML = this.getShowText(key, defaultValue);
    },
    /**
     * 初始化全局键盘监听
     */
    initGlobalKeyboardListener() {
      let localValue = this.getValue();
      if (!localValue.length) {
        /* 没有设置快捷键 */
        return;
      }
      utils.listenKeyboard(
        window,
        "keydown",
        (keyName, keyValue, ohterCodeList) => {
          if (NetDiskShortcut.isWaitUserPressKeyboard) {
            return;
          }
          localValue = this.getValue();
          let findShortcutIndex = localValue.findIndex((item) => {
            let itemValue = item["value"];
            let tempValue = {
              keyName: keyName,
              keyValue: keyValue,
              ohterCodeList: ohterCodeList,
            };
            if (JSON.stringify(itemValue) === JSON.stringify(tempValue)) {
              return item;
            }
          });
          if (findShortcutIndex != -1) {
            let findShortcut = localValue[findShortcutIndex];
            log.info(["调用快捷键", findShortcut]);
            if (findShortcut["key"] === "netdisk-keyboard-open-setting") {
              log.info("打开设置");
              NetDiskUI.suspension.showSettingView();
            } else if (
              findShortcut["key"] ===
              "netdisk-keyboard-open-history-matching-records"
            ) {
              log.info("打开历史匹配记录");
              NetDiskUI.netDiskHistoryMatch.show();
            } else if (
              findShortcut["key"] === "netdisk-keyboard-open-accesscode-rule"
            ) {
              log.info("打开访问码规则");
              NetDiskUI.accessCodeRule.show();
            } else if (
              findShortcut["key"] === "netdisk-keyboard-open-user-rule"
            ) {
              log.info("打开用户自定义规则");
              NetDiskCustomRules.showUI(false);
            } else if (
              findShortcut["key"] ===
              "netdisk-keyboard-open-proactively-recognize-text"
            ) {
              log.info("打开主动识别文本");
              NetDiskUI.matchPasteText.show();
            } else {
              log.error("还未配置调用函数");
            }
          }
        }
      );
    },
  };

  /** pops弹窗 */
  const NetDiskPops = {
    /**
     * 普通信息框
     * @param {PopsAlertDetails} details 配置
     * @param {{
     * PC: {
     *  width: string,
     *  height: string,
     * },
     * Mobile: {
     *  width: string,
     *  height: string
     * }
     * }} sizeConfig 大小配置
     * @returns {PopsCallResult}
     */
    alert(details, sizeConfig) {
      details = this.handleDetails(details, sizeConfig);
      return pops.alert(details);
    },
    /**
     * 询问框
     * @param {PopsConfirmDetails} details 配置
     * @param {?{
     * PC: {
     *  width: string,
     *  height: string,
     * },
     * Mobile: {
     *  width: string,
     *  height: string
     * }
     * }} sizeConfig 大小配置
     * @returns {PopsCallResult}
     */
    confirm(details, sizeConfig) {
      details = this.handleDetails(details, sizeConfig);
      return pops.confirm(details);
    },
    /**
     * 加载层
     * @param {PopsLoadingDetails} details 配置
     * @returns {PopsCallResult}
     */
    loading(details) {
      if (typeof details["animation"] === "undefined") {
        details["animation"] = GM_getValue(
          "popsAnimation",
          NetDiskUI.defaultAnimation
        );
      }
      if (typeof details["forbiddenScroll"] === "undefined") {
        details["forbiddenScroll"] = NetDiskUI.defaultForbiddenScroll;
      }
      return pops.loading(details);
    },
    /**
     * 输入框
     * @param {PopsPromptDetails} details 配置
     * @param {?{
     * PC: {
     *  width: string,
     *  height: string,
     * },
     * Mobile: {
     *  width: string,
     *  height: string
     * }
     * }} sizeConfig 大小配置
     * @returns {PopsCallResult}
     */
    prompt(details, sizeConfig) {
      details = this.handleDetails(details, sizeConfig);
      return pops.prompt(details);
    },
    /**
     * 文件夹
     * @param {PopsFolderDetails} details 配置
     * @returns {PopsCallResult}
     */
    folder(details, sizeConfig) {
      details = this.handleDetails(details, sizeConfig);
      details["sort"] = {
        name: GM_getValue("pops-folder-sort-name", NetDiskUI.defaultSortName),
        isDesc: GM_getValue(
          "pops-folder-sort-is-desc",
          NetDiskUI.defaultSortDesc
        ),
        callback(target, event, sortName, sortDesc) {
          GM_setValue("pops-folder-sort-name", sortName);
          GM_setValue("pops-folder-sort-is-desc", sortDesc);
        },
      };
      return pops.folder(details);
    },
    /**
     * 菜单面板
     * @param {PopsPanelDetails} details 配置
     * @returns {PopsCallResult}
     */
    panel(details, sizeConfig) {
      details = this.handleDetails(details, sizeConfig);
      return pops.panel(details);
    },
    /**
     *
     * @param {PopsConfirmDetails} details
     * @param {?{
     * PC: {
     *  width: string,
     *  height: string,
     * },
     * Mobile: {
     *  width: string,
     *  height: string
     * }
     * }} sizeConfig 大小配置
     */
    handleDetails(details, sizeConfig) {
      details = Object.assign(
        {
          animation: GM_getValue("popsAnimation", NetDiskUI.defaultAnimation),
          drag: GM_getValue("pcDrag", NetDiskUI.defaultPCDrag),
          dragLimit: GM_getValue("pcDragLimit", NetDiskUI.defaultPCDragLimit),
          forbiddenScroll: NetDiskUI.defaultForbiddenScroll,
        },
        details
      );
      if (sizeConfig != null) {
        details.width = pops.isPhone()
          ? sizeConfig.Mobile.width
          : sizeConfig.PC.width;
        details.height = pops.isPhone()
          ? sizeConfig.Mobile.height
          : sizeConfig.PC.height;
      }

      if (details.mask == null) {
        details.mask = {};
      }
      if (details.mask.enable == null) {
        details.mask.enable = true;
      }
      if (details.mask.clickEvent == null) {
        details.mask.clickEvent = {};
      }
      if (details.mask.clickEvent.toClose == null) {
        details.mask.clickEvent.toClose = GM_getValue(
          "clickMaskToCloseDialog",
          NetDiskUI.defaultClickMaskToCloseDialog
        );
      }
      if (GM_getValue("popsAcrylic", NetDiskUI.defaultPopsAcrylic)) {
        let acrylicCSS = `
        .pops {
          --acrylic-opacity: 0.7;
          --acrylic-color: rgba(232, 232, 232, var(--acrylic-opacity));
          --acrylic-blur: 30px;
          --acrylic-saturate: 125%;
          --pops-bg-opacity: var(--acrylic-opacity);
        }
        .pops {
          backdrop-filter: blur(var(--acrylic-blur)) saturate(var(--acrylic-saturate));
          background-color: var(--acrylic-color);
        }
        .pops[type-value=panel]{
          --aside-bg-color: rgba(221, 221, 221, var(--acrylic-opacity));
        }
        `;
        if (typeof details.style === "string") {
          details.style += acrylicCSS;
        } else {
          details.style = acrylicCSS;
        }
      }
      return details;
    },
  };

  /** 油猴菜单 */
  const NetDiskMenu = {
    /**
     * 初始化
     */
    init() {
      /* 不在iframe内进行注册 */
      if (!NetDiskUI.suspension.isTopWindow()) {
        return;
      }
      GM_Menu.add([
        {
          key: "showSetting",
          text: "⚙ 设置",
          showText(text) {
            return text;
          },
          callback() {
            NetDiskUI.suspension.showSettingView();
          },
        },
        {
          key: "showNetDiskHistoryMatch",
          text: "⚙ 历史匹配记录",
          showText(text) {
            return text;
          },
          callback() {
            NetDiskUI.netDiskHistoryMatch.show();
          },
        },
        {
          key: "showAccessCodeRule",
          text: "⚙ 访问码规则",
          showText(text) {
            return text;
          },
          callback() {
            NetDiskUI.accessCodeRule.show();
          },
        },
        {
          key: "showUserRule",
          text: "⚙ 用户自定义规则",
          showText(text) {
            return text;
          },
          callback() {
            NetDiskCustomRules.showUI(false);
          },
        },
        {
          key: "showMatchPasteText",
          text: "⚙ 主动识别文本",
          showText(text) {
            return text;
          },
          callback() {
            NetDiskUI.matchPasteText.show();
          },
        },
      ]);
    },
    /**
     * 初始化环境变量
     */
    initEnv() {
      if (GM_getValue("allowMatchLocationHref")) {
        NetDisk.allowMatchLocationHref = true;
      }
    },
  };

  /* -------入口------- */
  log.config({
    logMaxCount: 200000,
    autoClearConsole: false,
  });
  Object.assign(
    NetDiskUI.src.icon,
    typeof RESOURCE_ICON === "undefined" ? {} : RESOURCE_ICON
  );
  NetDiskMenu.init();
  NetDiskMenu.initEnv();
  NetDiskUI.initQmsg();
  NetDiskCustomRules.init();
  NetDisk.init();
  NetDiskShortcut.initGlobalKeyboardListener();
  DOMUtils.ready(function () {
    httpx.config({
      headers: {
        "User-Agent": utils.getRandomPCUA(),
      },
      onabort() {
        Qmsg.error("请求被取消");
      },
      ontimeout() {
        Qmsg.error("请求超时");
      },
      onerror(response) {
        Qmsg.error("请求异常");
        log.error(["httpx-onerror", response]);
      },
    });
    NetDiskAutoFillAccessCode.init();
    NetDiskAuthorization.init();
    NetDiskUI.monitorDOMInsert();
  });
})();