GreasyFork优化

自动登录账号、快捷寻找自己库被其他脚本引用、更新自己的脚本列表、库、优化图片浏览、美化页面、Markdown复制按钮

// ==UserScript==
// @name               GreasyFork优化
// @name:en-US         GreasyFork Optimization
// @namespace          https://github.com/WhiteSevs/TamperMonkeyScript
// @version            2024.7.20.11
// @author             WhiteSevs
// @description        自动登录账号、快捷寻找自己库被其他脚本引用、更新自己的脚本列表、库、优化图片浏览、美化页面、Markdown复制按钮
// @description:en-US  Automatically log in to the account, quickly find your own library referenced by other scripts, update your own script list, library, optimize image browsing, beautify the page, Markdown copy button
// @license            GPL-3.0-only
// @icon               
// @supportURL         https://github.com/WhiteSevs/TamperMonkeyScript/issues
// @match              *://greasyfork.org/*
// @require            https://update.greasyfork.org/scripts/494167/1413255/CoverUMD.js
// @require            https://fastly.jsdelivr.net/npm/qmsg@1.2.1/dist/index.umd.js
// @require            https://fastly.jsdelivr.net/npm/@whitesev/utils@1.7.0/dist/index.umd.js
// @require            https://fastly.jsdelivr.net/npm/@whitesev/domutils@1.1.2/dist/index.umd.js
// @require            https://fastly.jsdelivr.net/npm/@whitesev/pops@1.3.1/dist/index.umd.js
// @require            https://fastly.jsdelivr.net/npm/viewerjs@1.11.6/dist/viewer.min.js
// @require            https://fastly.jsdelivr.net/npm/i18next@23.12.2/i18next.min.js
// @resource           ViewerCSS  https://fastly.jsdelivr.net/npm/viewerjs@1.11.6/dist/viewer.min.css
// @connect            greasyfork.org
// @grant              GM_addStyle
// @grant              GM_deleteValue
// @grant              GM_getResourceText
// @grant              GM_getValue
// @grant              GM_info
// @grant              GM_registerMenuCommand
// @grant              GM_setValue
// @grant              GM_unregisterMenuCommand
// @grant              GM_xmlhttpRequest
// @grant              unsafeWindow
// @run-at             document-start
// ==/UserScript==

(t=>{function d(n){if(typeof n!="string")throw new TypeError("cssText must be a string");let e=document.createElement("style");return e.setAttribute("type","text/css"),e.innerHTML=n,document.head?document.head.appendChild(e):document.body?document.body.appendChild(e):document.documentElement.childNodes.length===0?document.documentElement.appendChild(e):document.documentElement.insertBefore(e,document.documentElement.childNodes[0]),e}if(typeof GM_addStyle=="function"){GM_addStyle(t);return}d(t)})(" .whitesev-hide{display:none}.whitesev-hide-important{display:none!important} ");

(function (Qmsg, DOMUtils, Utils, i18next, pops, Viewer) {
  'use strict';

  var _a;
  var _GM_addStyle = /* @__PURE__ */ (() => typeof GM_addStyle != "undefined" ? GM_addStyle : void 0)();
  var _GM_getResourceText = /* @__PURE__ */ (() => typeof GM_getResourceText != "undefined" ? GM_getResourceText : void 0)();
  var _GM_getValue = /* @__PURE__ */ (() => typeof GM_getValue != "undefined" ? GM_getValue : void 0)();
  var _GM_info = /* @__PURE__ */ (() => typeof GM_info != "undefined" ? GM_info : void 0)();
  var _GM_registerMenuCommand = /* @__PURE__ */ (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : void 0)();
  var _GM_setValue = /* @__PURE__ */ (() => typeof GM_setValue != "undefined" ? GM_setValue : void 0)();
  var _GM_unregisterMenuCommand = /* @__PURE__ */ (() => typeof GM_unregisterMenuCommand != "undefined" ? GM_unregisterMenuCommand : void 0)();
  var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)();
  var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)();
  var _monkeyWindow = /* @__PURE__ */ (() => window)();
  const zh_CN_language = {
    GreasyFork优化: "GreasyFork优化",
    请求取消: "请求取消",
    请求超时: "请求超时",
    请求异常: "请求异常",
    通用: "通用",
    账号: "账号",
    密码: "密码",
    语言: "语言",
    "账号/密码": "账号/密码",
    请输入账号: "请输入账号",
    请输入密码: "请输入密码",
    自动登录: "自动登录",
    自动登录当前保存的账号: "自动登录当前保存的账号",
    "清空账号/密码": "清空账号/密码",
    点击清空: "点击清空",
    "确定清空账号和密码?": "确定清空账号和密码?",
    "已清空账号/密码": "已清空账号/密码",
    "源代码同步【脚本列表】": "源代码同步【脚本列表】",
    一键同步: "一键同步",
    前往用户主页: "前往用户主页",
    获取当前已登录的用户主页失败: "获取当前已登录的用户主页失败",
    "源代码同步【未上架的脚本】": "源代码同步【未上架的脚本】",
    "源代码同步【库】": "源代码同步【库】",
    论坛: "论坛",
    功能: "功能",
    过滤重复的评论: "过滤重复的评论",
    "过滤掉重复的评论数量(≥2)": "过滤掉重复的评论数量(≥2)",
    "过滤脚本(id)": "过滤脚本(id)",
    "请输入脚本id,每行一个": "请输入脚本id,每行一个",
    "过滤发布的用户(id)": "过滤发布的用户(id)",
    "请输入用户id,每行一个": "请输入用户id,每行一个",
    "过滤回复的用户(id)": "过滤回复的用户(id)",
    优化: "优化",
    固定当前语言: "固定当前语言",
    无: "无",
    "如button、input、textarea": "如button、input、textarea",
    更直观的查看版本迭代: "更直观的查看版本迭代",
    美化上传图片按钮: "美化上传图片按钮",
    放大上传区域: "放大上传区域",
    优化图片浏览: "优化图片浏览",
    使用Viewer浏览图片: "使用Viewer浏览图片",
    覆盖图床图片跳转: "覆盖图床图片跳转",
    "配合上面的【优化图片浏览】更优雅浏览图片": "配合上面的【优化图片浏览】更优雅浏览图片",
    '需安装Greasyfork Beautify脚本,<a href="https://greasyfork.org/zh-CN/scripts/446849-greasyfork-beautify" target="_blank">🖐点我安装</a>': '需安装Greasyfork Beautify脚本,<a href="https://greasyfork.org/zh-CN/scripts/446849-greasyfork-beautify" target="_blank">🖐点我安装</a>',
    代码: "代码",
    添加复制代码按钮: "添加复制代码按钮",
    更优雅的复制: "更优雅的复制",
    快捷键: "快捷键",
    "【F】键全屏、【Alt+Shift+F】键宽屏": "【F】键全屏、【Alt+Shift+F】键宽屏",
    库: "库",
    脚本列表: "脚本列表",
    "请输入屏蔽规则,每行一个": "请输入屏蔽规则,每行一个",
    请求admin内容失败: "请求admin内容失败",
    解析admin的源代码同步表单失败: "解析admin的源代码同步表单失败",
    源代码同步失败: "源代码同步失败",
    获取用户信息失败: "获取用户信息失败",
    获取用户的收藏集失败: "获取用户的收藏集失败",
    "解析Script Sets失败": "解析Script Sets失败",
    "获取收藏集{{setsId}}失败": "获取收藏集{{setsId}}失败",
    "获取表单元素#edit_script_set失败": "获取表单元素#edit_script_set失败",
    更新收藏集表单请求失败: "更新收藏集表单请求失败",
    请先在菜单中录入账号: "请先在菜单中录入账号",
    请先在菜单中录入密码: "请先在菜单中录入密码",
    "获取csrf-token失败": "获取csrf-token失败",
    "正在登录中...": "正在登录中...",
    "登录失败,请在控制台查看原因": "登录失败,请在控制台查看原因",
    "登录成功,1s后自动跳转": "登录成功,1s后自动跳转",
    "登录失败,可能是账号/密码错误,请在控制台查看原因": "登录失败,可能是账号/密码错误,请在控制台查看原因",
    "美化 历史版本 页面": "美化 历史版本 页面",
    未找到history_versions元素列表: "未找到history_versions元素列表",
    "yyyy年MM月dd日 HH:mm:ss": "yyyy-MM-dd HH:mm:ss",
    "美化 Greasyfork Beautify脚本": "美化 Greasyfork Beautify脚本",
    "❌ 最多同时长传5张图": "❌ 最多同时长传5张图片",
    "❌ 图片:{{name}} 大小:{{size}}": "❌ 图片:{{name}} 大小:{{size}}",
    "已过滤:{{oldCount}}": "已过滤:{{oldCount}}",
    寻找引用: "寻找引用",
    获取脚本id失败: "获取脚本id失败",
    收藏: "收藏",
    请先登录账号: "请先登录账号",
    获取用户id失败: "获取用户id失败",
    "获取收藏夹中...": "获取收藏夹中...",
    收藏集: "收藏集",
    "添加中...": "添加中...",
    "添加失败,{{selector}}元素不存在": "添加失败,{{selector}}元素不存在",
    "未找到{{selector}}元素": "未找到{{selector}}元素",
    添加失败: "添加失败",
    添加成功: "添加成功",
    "删除中...": "删除中...",
    删除成功: "删除成功",
    添加: "添加",
    刪除: "刪除",
    "拦截跳转:": "拦截跳转:",
    今日检查: "今日检查",
    复制代码: "复制代码",
    "加载文件中...": "加载文件中...",
    复制成功: "复制成功",
    "✅ 复制成功!": "✅ 复制成功!",
    "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}": "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}",
    "导航至:": "导航至:",
    "请先登录账号!": "请先登录账号!",
    "获取信息中,请稍后...": "获取信息中,请稍后...",
    "获取成功,共 {{count}} 个": "获取成功,共 {{count}} 个",
    "评分:": "评分:",
    "语言:": "语言:",
    "版本:": "版本:",
    "更新:": "更新:",
    同步代码: "同步代码",
    "同步中...": "同步中...",
    手动: "手动",
    自动: "自动",
    "同步方式:{{syncMode}}": "同步方式:{{syncMode}}",
    同步成功: "同步成功",
    同步失败: "同步失败",
    该脚本未设置同步信息: "该脚本未设置同步信息",
    "上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载": "上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载",
    "名称:": "名称:",
    "进度:": "进度:",
    "未获取到【脚本列表】": "未获取到【脚本列表】",
    "源代码同步成功,3秒后更新下一个": "源代码同步成功,3秒后更新下一个",
    全部更新失败: "全部更新失败",
    "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}": "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}",
    "⚙ 设置": "⚙ 设置",
    "{{SCRIPT_NAME}}-设置": "{{SCRIPT_NAME}}-设置",
    美化页面元素: "美化页面元素",
    美化历史版本页面: "美化历史版本页面",
    "美化Greasyfork Beautify脚本": "美化Greasyfork Beautify脚本",
    获取表单csrfToken失败: "获取表单csrfToken失败",
    Toast配置: "Toast配置",
    Toast位置: "Toast位置",
    左上角: "左上角",
    顶部: "顶部",
    右上角: "右上角",
    左边: "左边",
    中间: "中间",
    右边: "右边",
    左下角: "左下角",
    底部: "底部",
    右下角: "右下角",
    Toast显示在页面九宫格的位置: "Toast显示在页面九宫格的位置",
    最多显示的数量: "最多显示的数量",
    限制Toast显示的数量: "限制Toast显示的数量",
    逆序弹出: "逆序弹出",
    修改Toast弹出的顺序: "修改Toast弹出的顺序",
    该脚本已经在该收藏集中: "该脚本已经在该收藏集中",
    其它错误: "其它错误",
    启用: "启用",
    开启后下面的过滤功能才会生效: "开启后下面的功能才会生效",
    屏蔽脚本: "屏蔽脚本",
    点击查看规则: "点击查看规则",
    过滤: "过滤",
    代码同步: "代码同步",
    美化: "美化",
    修复代码行号显示: "修复代码行号显示",
    修复代码行数超过1k行号显示不全问题: "修复代码行数超过1k行号显示不全问题",
    "添加【寻找引用】按钮": "添加【寻找引用】按钮",
    "在脚本栏添加按钮,一般用于搜索引用该库的相关脚本": "在脚本栏添加按钮,一般用于搜索引用该库的相关脚本",
    "添加【收藏】按钮": "添加【收藏】按钮",
    "在脚本栏添加按钮,一般用于快捷收藏该脚本/库": "在脚本栏添加按钮,一般用于快捷收藏该脚本/库",
    修复图片宽度显示问题: "修复图片宽度显示问题",
    修复图片在移动端宽度超出浏览器宽度问题: "修复图片在移动端宽度超出浏览器宽度问题",
    "添加【今日检查】信息块": "添加【今日检查】信息块",
    "在脚本信息栏添加【今日检查】信息块": "在脚本信息栏添加【今日检查】信息块",
    "给Markdown添加【复制】按钮": "给Markdown添加【复制】按钮",
    "在Markdown内容右上角添加【复制】按钮,点击一键复制Markdown内容": "在Markdown内容右上角添加【复制】按钮,点击一键复制Markdown内容",
    开启后下面的功能才会生效: "开启后下面的功能才会生效",
    检测页面加载: "检测页面加载",
    "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面": "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面",
    检测间隔: "检测间隔",
    "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面": "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面",
    美化顶部导航栏: "美化顶部导航栏",
    "可能会跟Greasyfork Beautify脚本有冲突": "可能会跟Greasyfork Beautify脚本有冲突",
    美化脚本列表: "美化脚本列表",
    "双列显示且添加脚本卡片操作项(安装、收藏)": "双列显示且添加脚本卡片操作项(安装、收藏)",
    操作面板: "操作面板",
    "添加【操作面板】按钮": "添加【操作面板】按钮",
    "在脚本列表页面时为顶部导航栏添加【操作面板】按钮": "在脚本列表页面时为顶部导航栏添加【操作面板】按钮",
    操作: "操作",
    安装此脚本: "安装此脚本",
    脚本: "脚本",
    历史版本: "历史版本",
    自定义已读颜色: "自定义已读颜色",
    在讨论内生效: "在讨论内生效",
    用户: "用户",
    控制台: "控制台",
    "迁移【控制台】到顶部导航栏": "迁移【控制台】到顶部导航栏",
    "将【控制台】按钮移动到顶部导航栏,节省空间": "将【控制台】按钮移动到顶部导航栏,节省空间",
    "在版本下面添加【安装】、【查看代码】按钮": "在版本下面添加【安装】、【查看代码】按钮",
    查看代码: "查看代码",
    添加快捷操作按钮: "添加快捷操作按钮",
    "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效": "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效",
    选择需要过滤的选项: "选择需要过滤的选项",
    "确定{{type}}:{{filterId}}?": "确定{{type}}:{{filterId}}?",
    "该收藏集未包含:{{scriptId}}": "该收藏集未包含:{{scriptId}}",
    帮助文档: "帮助文档",
    "请输入规则,每行一个": "请输入规则,每行一个",
    选择过滤的选项: "选择过滤的选项",
    "脚本id:{{text}}": "脚本id:{{text}}",
    "脚本名:{{text}}": "脚本名:{{text}}",
    "作者id:{{text}}": "作者id:{{text}}",
    "作者名:{{text}}": "作者名:{{text}}",
    "作用域:脚本、脚本搜索、用户主页": "作用域:脚本、脚本搜索、用户主页",
    "更新到 {{version}} 版本": "更新到 {{version}} 版本",
    "降级到 {{version}} 版本": "降级到 {{version}} 版本",
    "重新安装 {{version}} 版本": "重新安装 {{version}} 版本",
    "发布的用户id:{{text}}": "发布的用户id:{{text}}"
  };
  const en_US_language = {
    GreasyFork优化: "GreasyFork Optimization",
    请求取消: "http request cancel",
    请求超时: "http request timeout",
    请求异常: "http request error",
    通用: "General",
    账号: "Account",
    密码: "Password",
    语言: "Language",
    "账号/密码": "Account/Password",
    请输入账号: "Please enter your account number",
    请输入密码: "Please enter password",
    自动登录: "Auto Login",
    自动登录当前保存的账号: "Automatically log in to the currently saved account",
    "清空账号/密码": "Clear account/password",
    点击清空: "Clear",
    "确定清空账号和密码?": "Are you sure to clear your account and password?",
    "已清空账号/密码": "Account/password cleared",
    "源代码同步【脚本列表】": "Source Code Synchronization [Script List]",
    一键同步: "Sync All",
    前往用户主页: "Go to the user's homepage",
    获取当前已登录的用户主页失败: "Failed to retrieve the currently logged in user's homepage",
    "源代码同步【未上架的脚本】": "Source code synchronization [Script not listed]",
    "源代码同步【库】": "Source code synchronization 【 Library 】",
    论坛: "Forum",
    功能: "Function",
    过滤重复的评论: "Filter duplicate comments",
    "过滤掉重复的评论数量(≥2)": "Filter out duplicate comments (≥ 2)",
    "过滤脚本(id)": "Filter script (id)",
    "请输入脚本id,每行一个": "Please enter the script ID, one per line",
    "过滤发布的用户(id)": "Filter published users (id)",
    "请输入用户id,每行一个": "Please enter the user ID, one per line",
    "过滤回复的用户(id)": "User (ID) who filters replies",
    优化: "Optimization",
    固定当前语言: "Fix current language",
    无: "nothing",
    "如button、input、textarea": "For example button、input、textarea",
    更直观的查看版本迭代: "More intuitive viewing of version iterations",
    美化上传图片按钮: "Beautify upload image button",
    放大上传区域: "Enlarge the upload area",
    优化图片浏览: "Optimize image browsing",
    使用Viewer浏览图片: "Using Viewer to browse images",
    覆盖图床图片跳转: "Overlay bed image jump",
    "配合上面的【优化图片浏览】更优雅浏览图片": "Collaborate with the optimization of image browsing above to browse images more elegantly",
    '需安装Greasyfork Beautify脚本,<a href="https://greasyfork.org/zh-CN/scripts/446849-greasyfork-beautify" target="_blank">🖐点我安装</a>': 'Greasyfork Beauty script needs to be installed,<a href="https://greasyfork.org/zh-CN/scripts/446849-greasyfork-beautify" target="_blank">🖐 Click me to install</a>',
    代码: "Code",
    添加复制代码按钮: "Add Copy Code Button",
    更优雅的复制: "More elegant replication",
    快捷键: "Shortcut keys",
    "【F】键全屏、【Alt+Shift+F】键宽屏": "【F】 Key full screen, [Alt+Shift+F] key wide screen",
    库: "Library",
    脚本列表: "Script List",
    "请输入屏蔽规则,每行一个": "Please enter a blocking rule, one per line",
    请求admin内容失败: "Request for admin content failed",
    解析admin的源代码同步表单失败: "Failed to parse the source code of admin and synchronize the form",
    源代码同步失败: "Source code synchronization failed",
    获取用户信息失败: "Failed to obtain user information",
    获取用户的收藏集失败: "Failed to retrieve user's collection",
    "解析Script Sets失败": "Parsing Script Sets failed",
    "获取收藏集{{setsId}}失败": "Failed to retrieve collection {{setsId}}",
    "获取表单元素#edit_script_set失败": "Failed to retrieve form element #edit_script_set",
    更新收藏集表单请求失败: "Update collection form request failed",
    请先在菜单中录入账号: "Please enter your account in the menu first",
    请先在菜单中录入密码: "Please enter your password in the menu first",
    "获取csrf-token失败": "Failed to obtain csrf token",
    "正在登录中...": "Logging in...",
    "登录失败,请在控制台查看原因": "Login failed, please check the reason in the console",
    "登录成功,1s后自动跳转": "Login successful, automatically redirect after 1 second",
    "登录失败,可能是账号/密码错误,请在控制台查看原因": "Login failed, possibly due to incorrect account/password. Please check the reason in the console",
    "美化 历史版本 页面": "Beautify the historical version page",
    未找到history_versions元素列表: "History_versions element list not found",
    "yyyy年MM月dd日 HH:mm:ss": "yyyy-MM-dd HH:mm:ss",
    "美化 Greasyfork Beautify脚本": "Beautify Greasyfork Beauty Script",
    "❌ 最多同时长传5张图": "❌ Upload up to 5 images simultaneously",
    "❌ 图片:{{name}} 大小:{{size}}": "❌ Image:{{name}} Size:{{size}}",
    "已过滤:{{oldCount}}": "Filtered:{{oldCount}}",
    寻找引用: "Find references",
    获取脚本id失败: "Failed to obtain script ID",
    收藏: "Collection",
    请先登录账号: "Please log in to your account first",
    获取用户id失败: "Failed to obtain user ID",
    "获取收藏夹中...": "Get in favorites...",
    收藏集: "Collection",
    "添加中...": "Adding...",
    "添加失败,{{selector}}元素不存在": "Add failed, {{selector}} element does not exist",
    "未找到{{selector}}元素": "{{selector}} element not found",
    添加失败: "Add failed",
    添加成功: "Successfully added",
    "删除中...": "Deleting...",
    删除成功: "Delete successful",
    添加: "Add in deletion",
    刪除: "Delete",
    "拦截跳转:": "Intercept jump:",
    今日检查: "Today's inspection",
    复制代码: "Copy Code",
    "加载文件中...": "Loading files...",
    复制成功: "Copy successful",
    "✅ 复制成功!": "✅ Copy successful!",
    "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}": "Current language: {{currentLocaleLanguage}}, switch to {{localeLanguage}} in 3 seconds",
    "导航至:": "Navigation to:",
    "请先登录账号!": "Please log in to your account first!",
    "获取信息中,请稍后...": "Obtaining information, please wait...",
    "获取成功,共 {{count}} 个": "Successfully obtained, a total of {{count}}",
    "评分:": "Rating:",
    "语言:": "Language:",
    "版本:": "Version:",
    "更新:": "Update:",
    同步代码: "Synchronize Code",
    "同步中...": "Synchronizing...",
    手动: "Manual",
    自动: "Automatic",
    "同步方式:{{syncMode}}": "Synchronization method: {{syncMode}}",
    同步成功: "Sync successful",
    同步失败: "Sync failed",
    该脚本未设置同步信息: "The script has not set synchronization information",
    "上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载": "Last reload time {{time}}, rejected repeated reloads within {{timeout}} seconds",
    "名称:": "Name:",
    "进度:": "Progress:",
    "未获取到【脚本列表】": "Unable to obtain [Script List]",
    "源代码同步成功,3秒后更新下一个": "Source code synchronization successful, update next one in 3 seconds",
    全部更新失败: "All updates failed",
    "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}": "All updates completed<br>Success: {{successNums}}<br>Failure: {{failed Nums}}<br>Total: {{scriptUrlListLength}}",
    "⚙ 设置": "⚙  Setting",
    "{{SCRIPT_NAME}}-设置": "{{SCRIPT_NAME}}-Setting",
    美化页面元素: "Beautify page elements",
    美化历史版本页面: "Beautify the historical version page",
    "美化Greasyfork Beautify脚本": "Beautify Greasyfork Beauty Script",
    获取表单csrfToken失败: "Failed to obtain form csrfToken",
    Toast配置: "Toast Config",
    Toast位置: "Toast position",
    左上角: "Top left",
    顶部: "Top",
    右上角: "Top right",
    左边: "Left",
    中间: "Center",
    右边: "Right",
    左下角: "Bottom left",
    底部: "Bottom",
    右下角: "Bottom right",
    Toast显示在页面九宫格的位置: "Toast is displayed in the nine grid position on the page",
    最多显示的数量: "Maximum number of displays",
    限制Toast显示的数量: "Limit the number of Toast displays",
    逆序弹出: "Reverse pop-up",
    修改Toast弹出的顺序: "Modify the order in which Toast pops up",
    该脚本已经在该收藏集中: "The script is already in this collection",
    其它错误: "Ohter Error",
    启用: "Enable",
    开启后下面的过滤功能才会生效: "The following filtering function will only take effect after it is enabled",
    屏蔽脚本: "Block script",
    点击查看规则: "Click to view rules",
    过滤: "Filter",
    代码同步: "Code synchronization",
    美化: "Beautify",
    修复代码行号显示: "Fix code line number display",
    修复代码行数超过1k行号显示不全问题: "Fix the problem that the code line number display is not complete when the number of lines exceeds 1k",
    "添加【寻找引用】按钮": "Add the button to find references",
    "在脚本栏添加按钮,一般用于搜索引用该库的相关脚本": "Add a button to the script bar, generally used to search for scripts that reference this library",
    "添加【收藏】按钮": "Add the button to collect",
    "在脚本栏添加按钮,一般用于快捷收藏该脚本/库": "Add a button to the script bar, generally used to quickly collect this script / library",
    修复图片宽度显示问题: " Fix the problem that the picture width display is not complete",
    修复图片在移动端宽度超出浏览器宽度问题: "Fix the problem that the picture width exceeds the browser width on mobile",
    "添加【今日检查】信息块": "Add the block of information of today's inspection",
    "在脚本信息栏添加【今日检查】信息块": "Add the block of information of today's inspection to the script information bar",
    "给Markdown添加【复制】按钮": "Add the button to copy to Markdown",
    "在Markdown内容右上角添加【复制】按钮,点击一键复制Markdown内容": "Add the button to copy to the top right corner of the Markdown content, click to copy the Markdown content in one click",
    开启后下面的功能才会生效: "The following functions will only take effect after it is enabled",
    检测页面加载: "Detect page loading",
    "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面": "Detect whether the Greasyfork page is loaded normally. If the loading fails, the page will be automatically refreshed",
    检测间隔: "Detection interval",
    "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面": "Set the interval time for detecting the last refresh page. If the time since the last refresh page exceeds the set value, the page will no longer be refreshed",
    美化顶部导航栏: "Beautify the top navigation bar",
    "可能会跟Greasyfork Beautify脚本有冲突": "Possible conflict with Greasymfork Beautify script",
    美化脚本列表: "Beautify Script List",
    "双列显示且添加脚本卡片操作项(安装、收藏)": "Double column display and add script card operation items (installation, bookmarking)",
    操作面板: "Operation Panel",
    "添加【操作面板】按钮": "Add [Operation Panel] button",
    "在脚本列表页面时为顶部导航栏添加【操作面板】按钮": "Add an 'Operation Panel' button to the top navigation bar on the script list page",
    操作: "Operation",
    安装此脚本: "Install this script",
    脚本: "Scripts",
    历史版本: "Historical version",
    自定义已读颜色: "Customize read colors",
    在讨论内生效: "Effective within the discussion",
    用户: "Users",
    控制台: "Console",
    "迁移【控制台】到顶部导航栏": "Migration of Console to Top Navigation Bar",
    "将【控制台】按钮移动到顶部导航栏,节省空间": "Move the 'Console' button to the top navigation bar to save space",
    添加额外的标签按钮: "Add additional label button",
    "在版本下面添加【安装】、【查看代码】按钮": "Add 【 Install 】 and 【 View Code 】 buttons under the version",
    查看代码: "View Code",
    添加快捷操作按钮: "Add shortcut operation button",
    "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效": "Add a 'Filter' button at the end of each discussion line. The filtering function needs to be enabled for it to take effect",
    选择需要过滤的选项: "Select the options that need to be filtered",
    "确定{{type}}:{{filterId}}?": "Are you sure {{type}}:{{filterId}}?",
    "该收藏集未包含:{{scriptId}}": "This collection does not include:{{scriptId}}",
    帮助文档: "Help document",
    "请输入规则,每行一个": "Please enter a rule, one per line",
    选择过滤的选项: "Select filtering options",
    "脚本id:{{text}}": "Script Id: {{text}}",
    "脚本名:{{text}}": "Script Name: {{text}}",
    "作者id:{{text}}": "Author Id: {{text}}",
    "作者名:{{text}}": "Author Name: {{text}}",
    "作用域:脚本、脚本搜索、用户主页": "Scope: Script, Script Search, User Homepage",
    "更新到 {{version}} 版本": "Update To {{version}} Version",
    "降级到 {{version}} 版本": "Downgrade to {{version}} Version",
    "重新安装 {{version}} 版本": "Reinstall {{version}} Version",
    "发布的用户id:{{text}}": "Published user ID: {{text}}"
  };
  const KEY = "GM_Panel";
  const ATTRIBUTE_KEY = "data-key";
  const ATTRIBUTE_DEFAULT_VALUE = "data-default-value";
  const LanguageInit = function() {
    let settingPanel = _GM_getValue(KEY, {});
    let lng = settingPanel["setting-language"] || "zh-CN";
    i18next.init({
      lng,
      // lng: "zh-CN",
      fallbackLng: "zh-CN",
      resources: {
        "zh-CN": {
          translation: { ...zh_CN_language }
        },
        "en-US": {
          translation: { ...en_US_language }
        }
      }
    });
  };
  LanguageInit();
  _GM_getValue(KEY, {});
  const _SCRIPT_NAME_ = i18next.t("GreasyFork优化");
  const utils = Utils.noConflict();
  const domUtils = DOMUtils.noConflict();
  const __pops = pops;
  const log = new utils.Log(
    _GM_info,
    _unsafeWindow.console || _monkeyWindow.console
  );
  const SCRIPT_NAME = ((_a = _GM_info == null ? void 0 : _GM_info.script) == null ? void 0 : _a.name) || _SCRIPT_NAME_;
  const DEBUG = false;
  log.config({
    debug: DEBUG,
    logMaxCount: 1e3,
    autoClearConsole: true,
    tag: true
  });
  Qmsg.config(
    Object.defineProperties(
      {
        html: true,
        autoClose: true,
        showClose: false
      },
      {
        position: {
          get() {
            return PopsPanel.getValue("qmsg-config-position", "bottom");
          }
        },
        maxNums: {
          get() {
            return PopsPanel.getValue("qmsg-config-maxnums", 5);
          }
        },
        showReverse: {
          get() {
            return PopsPanel.getValue("qmsg-config-showreverse", true);
          }
        },
        zIndex: {
          get() {
            let maxZIndex = Utils.getMaxZIndex();
            let popsMaxZIndex = pops.config.InstanceUtils.getPopsMaxZIndex(maxZIndex).zIndex;
            return Utils.getMaxValue(maxZIndex, popsMaxZIndex) + 100;
          }
        }
      }
    )
  );
  const GM_Menu = new utils.GM_Menu({
    GM_getValue: _GM_getValue,
    GM_setValue: _GM_setValue,
    GM_registerMenuCommand: _GM_registerMenuCommand,
    GM_unregisterMenuCommand: _GM_unregisterMenuCommand
  });
  const httpx = new utils.Httpx(_GM_xmlhttpRequest);
  httpx.interceptors.response.use(void 0, (data) => {
    log.error(["拦截器-请求错误", data]);
    if (data.type === "onabort") {
      Qmsg.warning(i18next.t("请求取消"));
    } else if (data.type === "onerror") {
      Qmsg.error(i18next.t("请求异常"));
    } else if (data.type === "ontimeout") {
      Qmsg.error(i18next.t("请求超时"));
    } else {
      Qmsg.error(i18next.t("其它错误"));
    }
    return data;
  });
  httpx.config({
    logDetails: DEBUG
  });
  ({
    Object: {
      defineProperty: _unsafeWindow.Object.defineProperty
    },
    Function: {
      apply: _unsafeWindow.Function.prototype.apply,
      call: _unsafeWindow.Function.prototype.call
    },
    Element: {
      appendChild: _unsafeWindow.Element.prototype.appendChild
    },
    setTimeout: _unsafeWindow.setTimeout
  });
  const addStyle = utils.addStyle;
  const UIButton = function(text, description, buttonText, buttonIcon, buttonIsRightIcon, buttonIconIsLoading, buttonType, clickCallBack) {
    let result = {
      text,
      type: "button",
      description,
      buttonIcon,
      buttonIsRightIcon,
      buttonIconIsLoading,
      buttonType,
      buttonText,
      callback(event) {
        if (typeof clickCallBack === "function") {
          clickCallBack(event);
        }
      },
      afterAddToUListCallBack: void 0
    };
    return result;
  };
  const UIInput = function(text, key, defaultValue, description, changeCallBack, placeholder = "", isNumber, isPassword) {
    let result = {
      text,
      type: "input",
      isNumber: Boolean(isNumber),
      isPassword: Boolean(isPassword),
      attributes: {},
      description,
      getValue() {
        let localValue = PopsPanel.getValue(key, defaultValue);
        return localValue;
      },
      callback(event, value) {
        PopsPanel.setValue(key, value);
      },
      placeholder
    };
    if (result.attributes) {
      result.attributes[ATTRIBUTE_KEY] = key;
      result.attributes[ATTRIBUTE_DEFAULT_VALUE] = defaultValue;
    }
    return result;
  };
  const UISwitch = function(text, key, defaultValue, clickCallBack, description) {
    let result = {
      text,
      type: "switch",
      description,
      attributes: {},
      getValue() {
        return Boolean(PopsPanel.getValue(key, defaultValue));
      },
      callback(event, value) {
        log.success(`${value ? "开启" : "关闭"} ${text}`);
        PopsPanel.setValue(key, Boolean(value));
      },
      afterAddToUListCallBack: void 0
    };
    if (result.attributes) {
      result.attributes[ATTRIBUTE_KEY] = key;
      result.attributes[ATTRIBUTE_DEFAULT_VALUE] = Boolean(defaultValue);
    }
    return result;
  };
  const GreasyforkApi = {
    /**
     * 获取代码搜索地址
     * @param url
     */
    getCodeSearchUrl(url) {
      return "https://greasyfork.org/zh-CN/scripts/code-search?c=" + url;
    },
    /**
     * 获取管理地址
     * @param url
     */
    getAdminUrl(url) {
      return url + "/admin";
    },
    /**
     * 从字符串中提取Id
     * @param text
     */
    getScriptId(text) {
      var _a2, _b;
      return (_b = (_a2 = text || window.location.pathname) == null ? void 0 : _a2.match(
        /\/scripts\/([\d]+)/i
      )) == null ? void 0 : _b[1];
    },
    /**
     * 从字符串中提取用户id
     * @param text
     */
    getUserId(text) {
      var _a2;
      return (_a2 = (text || window.location.pathname).match(/\/users\/([\d]+)/i)) == null ? void 0 : _a2[1];
    },
    /**
     * 从字符串中提取脚本名
     * @param text
     */
    getScriptName(text) {
      let pathname = window.location.pathname;
      if (text != null) {
        pathname = new URL(text).pathname;
      }
      pathname = decodeURIComponent(pathname);
      let pathnameSplit = pathname.split("/");
      for (const name of pathnameSplit) {
        let nameMatch = name.match(/[\d]+/);
        if (nameMatch && nameMatch.length) {
          return nameMatch[1];
        }
      }
    },
    /**
     * 获取需要切换语言的Url
     */
    getSwitchLanguageUrl(localeLanguage = "zh-CN") {
      let url = window.location.origin;
      let urlSplit = window.location.pathname.split("/");
      urlSplit[1] = localeLanguage;
      url = url + urlSplit.join("/");
      url += window.location.search;
      if (window.location.search === "") {
        url += "?locale_override=1";
      } else if (!window.location.search.includes("locale_override=1")) {
        url += "&locale_override=1";
      }
      return url;
    },
    /**
     * 获取脚本统计数据
     * @param scriptId
     */
    async getScriptStats(scriptId) {
      return new Promise(async (resolve) => {
        let scriptStatsRequest = await httpx.get({
          url: `https://greasyfork.org/scripts/${scriptId}/stats.json`,
          fetch: true,
          onerror() {
          },
          ontimeout() {
          }
        });
        if (!scriptStatsRequest.status) {
          resolve(null);
          return;
        }
        let scriptStatsJSON = scriptStatsRequest.data;
        resolve(scriptStatsJSON);
      });
    },
    /**
     * 解析并获取admin内的源代码同步的配置表单
     * @param scriptId
     */
    async getSourceCodeSyncFormData(scriptId) {
      let getResp = await httpx.get(
        `https://greasyfork.org/zh-CN/scripts/${scriptId}/admin`,
        {
          fetch: true
        }
      );
      log.success(getResp);
      if (!getResp.status) {
        Qmsg.error(i18next.t("请求admin内容失败"));
        return;
      }
      let adminHTML = getResp.data.responseText;
      let adminHTMLElement = domUtils.parseHTML(adminHTML, false, true);
      let formElement = adminHTMLElement.querySelector("form.edit_script");
      if (!formElement) {
        Qmsg.error(i18next.t("解析admin的源代码同步表单失败"));
        return;
      }
      let formData = new FormData(formElement);
      return formData;
    },
    /**
     * 进行源代码同步,要求先getSourceCodeSyncFormData
     * @param scriptId
     * @param data
     */
    async sourceCodeSync(scriptId, data) {
      let postResp = await httpx.post(
        `https://greasyfork.org/zh-CN/scripts/${scriptId}/sync_update`,
        {
          fetch: true,
          data
        }
      );
      log.success(postResp);
      if (!postResp.status) {
        Qmsg.error(i18next.t("源代码同步失败"));
        return;
      }
      return postResp;
    },
    /**
     * 获取用户的信息,包括脚本列表、未上架的脚本、库
     */
    async getUserInfo(userId) {
      let getResp = await httpx.get(
        `https://greasyfork.org/zh-CN/users/${userId}.json`,
        {
          fetch: true
        }
      );
      log.success(getResp);
      if (!getResp.status) {
        Qmsg.error(i18next.t("获取用户信息失败"));
        return;
      }
      let data = utils.toJSON(getResp.data.responseText);
      data["scriptList"] = [];
      data["scriptLibraryList"] = [];
      data["scripts"].forEach((scriptInfo) => {
        if (scriptInfo["code_url"].endsWith(".user.js")) {
          data["scriptList"].push(scriptInfo);
        } else {
          data["scriptLibraryList"].push(scriptInfo);
        }
      });
      return data;
    },
    /**
     * 获取用户的收藏集
     * @param userId
     */
    async getUserCollection(userId) {
      let getResp = await httpx.get(
        `https://greasyfork.org/zh-CN/users/${userId}`,
        {
          fetch: true
        }
      );
      log.info(["获取用户的收藏集", getResp]);
      if (!getResp.status) {
        Qmsg.error(i18next.t("获取用户的收藏集失败"));
        return;
      }
      let respText = getResp.data.responseText;
      let respDocument = domUtils.parseHTML(respText, true, true);
      let userScriptSets = respDocument.querySelector("#user-script-sets");
      if (!userScriptSets) {
        log.error("解析Script Sets失败");
        return;
      }
      let scriptSetsIdList = [];
      userScriptSets.querySelectorAll("li").forEach((liElement) => {
        var _a2;
        let $ele = liElement.querySelector("a:last-child");
        if (!$ele) {
          return;
        }
        let setsUrl = $ele.href;
        if (setsUrl.includes("?fav=1")) {
          return;
        }
        let setsName = liElement.querySelector("a").innerText;
        let setsId = (_a2 = setsUrl.match(/\/sets\/([\d]+)\//)) == null ? void 0 : _a2[1];
        scriptSetsIdList.push({
          id: setsId,
          name: setsName
        });
      });
      return scriptSetsIdList;
    },
    /**
     * 获取某个收藏集的信息
     * @param userId 用户id
     * @param setsId 收藏集id
     */
    async getUserCollectionInfo(userId, setsId) {
      let getResp = await httpx.get(
        `https://greasyfork.org/zh-CN/users/${userId}/sets/${setsId}/edit`,
        {
          fetch: true
        }
      );
      if (!getResp.status) {
        Qmsg.error(i18next.t("获取收藏集{{setsId}}失败", { setsId }));
        return;
      }
      let respText = getResp.data.responseText;
      let respDocument = domUtils.parseHTML(respText, true, true);
      let $edit_script_set_form = respDocument.querySelector(
        'form[id^="edit_script_set"]'
      );
      if (!$edit_script_set_form) {
        Qmsg.error(i18next.t("获取表单元素#edit_script_set失败"));
        return;
      }
      let formData = new FormData($edit_script_set_form);
      let csrfToken = respDocument.querySelector(
        'meta[name="csrf-token"]'
      );
      if (!csrfToken) {
        Qmsg.error(i18next.t("获取表单csrfToken失败"));
        return;
      }
      if (csrfToken.hasAttribute("content")) {
        let authenticity_token = csrfToken.getAttribute("content");
        if (authenticity_token) {
          formData.set("authenticity_token", authenticity_token);
        }
      }
      return formData;
    },
    /**
     * 更新用户的某个收藏集的表单信息
     * @param userId 用户id
     * @param setsId 收藏集id
     * @param data
     */
    async updateUserSetsInfo(userId, setsId, data) {
      let postResp = await httpx.post(
        `https://greasyfork.org/zh-CN/users/${userId}/sets/${setsId}`,
        {
          fetch: true,
          headers: {
            accept: "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7",
            "accept-language": "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6",
            "cache-control": "no-cache",
            "content-type": "application/x-www-form-urlencoded",
            pragma: "no-cache"
          },
          fetchInit: {
            referrerPolicy: "strict-origin-when-cross-origin"
          },
          data
        }
      );
      if (!postResp.status) {
        Qmsg.error(i18next.t("更新收藏集表单请求失败"));
        return;
      }
      let respText = postResp.data.responseText;
      let respDocument = domUtils.parseHTML(respText, true, true);
      return respDocument;
    },
    /**
     * 切换语言
     * @param url
     */
    async switchLanguage(url) {
      let getResp = await httpx.get(url, {
        fetch: true,
        headers: {
          "Upgrade-Insecure-Requests": "1"
        }
      });
      if (!getResp.status) {
        return;
      }
      log.info(getResp);
    }
  };
  const GreasyforkRouter = {
    /**
     * 代码页面
     */
    isCode() {
      var _a2;
      return (_a2 = window.location.pathname.split("/")) == null ? void 0 : _a2.includes("code");
    },
    /**
     * (严格比较)代码页面
     */
    isCodeStrict() {
      return window.location.pathname.endsWith("/code");
    },
    /**
     * 版本页面
     */
    isVersion() {
      return window.location.pathname.endsWith("/versions");
    },
    /**
     * 用户主页
     */
    isUserHome() {
      return window.location.pathname.match(/\/.+\/users\/.+/gi);
    },
    /**
     * 脚本页面(单个脚本的页面)
     */
    isScript() {
      return window.location.pathname.includes("/scripts/");
    },
    /**
     * 脚本列表页面
     */
    isScriptList() {
      return window.location.pathname.endsWith("/scripts");
    },
    /**
     * 库列表页面
     */
    isScriptLibraryList() {
      return window.location.pathname.endsWith("/libraries");
    },
    /**
     * 脚本代码搜索页面
     */
    isScriptCodeSearch() {
      return window.location.pathname.endsWith("/code-search");
    },
    /**
     * 讨论页面
     */
    isDiscuessions() {
      return window.location.pathname.endsWith("/discussions");
    }
  };
  const GreasyforkMenu = {
    /**
     * @class
     */
    menu: GM_Menu,
    /**
     * 当前是否已登录
     */
    isLogin: false,
    /**
     * 初始化环境变量
     */
    initEnv() {
      let userLinkElement = this.getUserLinkElement();
      this.isLogin = Boolean(userLinkElement);
    },
    /**
     * 获取当前登录用户的a标签元素
     */
    getUserLinkElement() {
      return document.querySelector(
        "#nav-user-info span.user-profile-link a"
      );
    },
    /**
     * 更新脚本
     * @param scriptUrlList
     */
    async updateScript(scriptUrlList) {
      let getLoadingHTML = function(scriptName, progress = 1) {
        return `
			<div style="display: flex;flex-direction: column;align-items: flex-start;">
				<div style="height: 30px;line-height: 30px;">${i18next.t(
        "名称:"
      )}${scriptName}</div>
				<div style="height: 30px;line-height: 30px;">${i18next.t(
        "进度:"
      )}${progress}/${scriptUrlList.length}</div>
			</div>`;
      };
      if (utils.isNull(scriptUrlList)) {
        Qmsg.error(i18next.t("未获取到【脚本列表】"));
      } else {
        let loading = Qmsg.loading(
          getLoadingHTML(GreasyforkApi.getScriptName(scriptUrlList[0])),
          {
            html: true
          }
        );
        let successNums = 0;
        let failedNums = 0;
        for (let index = 0; index < scriptUrlList.length; index++) {
          let scriptUrl = scriptUrlList[index];
          let scriptId = GreasyforkApi.getScriptId(scriptUrl);
          log.success("更新:" + scriptUrl);
          let scriptName = GreasyforkApi.getScriptName(scriptUrl);
          loading.setHTML(getLoadingHTML(scriptName, index + 1));
          let codeSyncFormData = await GreasyforkApi.getSourceCodeSyncFormData(
            scriptId
          );
          if (codeSyncFormData) {
            let syncUpdateStatus = await GreasyforkApi.sourceCodeSync(
              scriptId,
              codeSyncFormData
            );
            if (syncUpdateStatus) {
              Qmsg.success(i18next.t("源代码同步成功,3秒后更新下一个"));
              await utils.sleep(3e3);
              successNums++;
            } else {
              Qmsg.error(i18next.t("源代码同步失败"));
              failedNums++;
            }
          } else {
            Qmsg.error(i18next.t("源代码同步失败"));
            failedNums++;
          }
        }
        loading.close();
        if (successNums === 0) {
          Qmsg.error(i18next.t("全部更新失败"));
        } else {
          Qmsg.success(
            i18next.t(
              "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}",
              {
                successNums,
                failedNums,
                scriptUrlListLength: scriptUrlList.length
              }
            ),
            {
              html: true
            }
          );
        }
      }
    },
    /**
     * 处理本地的goto事件
     */
    handleLocalGotoCallBack() {
      if (PopsPanel.getValue("goto_updateSettingsAndSynchronize_scriptList")) {
        PopsPanel.deleteValue("goto_updateSettingsAndSynchronize_scriptList");
        if (!GreasyforkRouter.isUserHome()) {
          PopsPanel.setValue(
            "goto_updateSettingsAndSynchronize_scriptList",
            true
          );
          if (GreasyforkMenu.getUserLinkElement()) {
            Qmsg.success(i18next.t("前往用户主页"));
            window.location.href = GreasyforkMenu.getUserLinkElement().href;
          } else {
            Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
          }
          return;
        }
        let scriptUrlList = [];
        document.querySelectorAll(
          "#user-script-list-section li a.script-link"
        ).forEach((item) => {
          scriptUrlList = scriptUrlList.concat(
            GreasyforkApi.getAdminUrl(item.href)
          );
        });
        GreasyforkMenu.updateScript(scriptUrlList);
      } else if (PopsPanel.getValue("goto_updateSettingsAndSynchronize_unlistedScriptList")) {
        PopsPanel.deleteValue(
          "goto_updateSettingsAndSynchronize_unlistedScriptList"
        );
        if (!GreasyforkRouter.isUserHome()) {
          PopsPanel.setValue(
            "goto_updateSettingsAndSynchronize_unlistedScriptList",
            true
          );
          if (GreasyforkMenu.getUserLinkElement()) {
            Qmsg.success(i18next.t("前往用户主页"));
            window.location.href = GreasyforkMenu.getUserLinkElement().href;
          } else {
            Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
          }
          return;
        }
        let scriptUrlList = [];
        document.querySelectorAll(
          "#user-unlisted-script-list li a.script-link"
        ).forEach((item) => {
          scriptUrlList = scriptUrlList.concat(
            GreasyforkApi.getAdminUrl(item.href)
          );
        });
        GreasyforkMenu.updateScript(scriptUrlList);
      } else if (PopsPanel.getValue("goto_updateSettingsAndSynchronize_libraryScriptList")) {
        PopsPanel.deleteValue(
          "goto_updateSettingsAndSynchronize_libraryScriptList"
        );
        if (!GreasyforkRouter.isUserHome()) {
          PopsPanel.setValue(
            "goto_updateSettingsAndSynchronize_libraryScriptList",
            true
          );
          if (GreasyforkMenu.getUserLinkElement()) {
            Qmsg.success(i18next.t("前往用户主页"));
            window.location.href = GreasyforkMenu.getUserLinkElement().href;
          } else {
            Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
          }
          return;
        }
        let scriptUrlList = [];
        document.querySelectorAll(
          "#user-library-script-list li a.script-link"
        ).forEach((item) => {
          scriptUrlList = scriptUrlList.concat(
            GreasyforkApi.getAdminUrl(item.href)
          );
        });
        GreasyforkMenu.updateScript(scriptUrlList);
      }
    }
  };
  const UISelect = function(text, key, defaultValue, data, callback, description) {
    let selectData = [];
    if (typeof data === "function") {
      selectData = data();
    } else {
      selectData = data;
    }
    let result = {
      text,
      type: "select",
      description,
      attributes: {},
      getValue() {
        return PopsPanel.getValue(key, defaultValue);
      },
      callback(event, isSelectedValue, isSelectedText) {
        PopsPanel.setValue(key, isSelectedValue);
        if (typeof callback === "function") {
          callback(event, isSelectedValue, isSelectedText);
        }
      },
      data: selectData
    };
    if (result.attributes) {
      result.attributes[ATTRIBUTE_KEY] = key;
      result.attributes[ATTRIBUTE_DEFAULT_VALUE] = defaultValue;
    }
    return result;
  };
  const SettingUIGeneral = {
    id: "greasy-fork-panel-config-account",
    title: i18next.t("通用"),
    forms: [
      {
        text: "",
        type: "forms",
        forms: [
          {
            text: i18next.t("Toast配置"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  UISelect(
                    i18next.t("Toast位置"),
                    "qmsg-config-position",
                    "bottom",
                    [
                      {
                        value: "topleft",
                        text: i18next.t("左上角")
                      },
                      {
                        value: "top",
                        text: i18next.t("顶部")
                      },
                      {
                        value: "topright",
                        text: i18next.t("右上角")
                      },
                      {
                        value: "left",
                        text: i18next.t("左边")
                      },
                      {
                        value: "center",
                        text: i18next.t("中间")
                      },
                      {
                        value: "right",
                        text: i18next.t("右边")
                      },
                      {
                        value: "bottomleft",
                        text: i18next.t("左下角")
                      },
                      {
                        value: "bottom",
                        text: i18next.t("底部")
                      },
                      {
                        value: "bottomright",
                        text: i18next.t("右下角")
                      }
                    ],
                    (event, isSelectValue, isSelectText) => {
                      log.info("设置当前Qmsg弹出位置" + isSelectText);
                    },
                    i18next.t("Toast显示在页面九宫格的位置")
                  ),
                  UISelect(
                    i18next.t("最多显示的数量"),
                    "qmsg-config-maxnums",
                    3,
                    [
                      {
                        value: 1,
                        text: "1"
                      },
                      {
                        value: 2,
                        text: "2"
                      },
                      {
                        value: 3,
                        text: "3"
                      },
                      {
                        value: 4,
                        text: "4"
                      },
                      {
                        value: 5,
                        text: "5"
                      }
                    ],
                    void 0,
                    i18next.t("限制Toast显示的数量")
                  ),
                  UISwitch(
                    i18next.t("逆序弹出"),
                    "qmsg-config-showreverse",
                    false,
                    void 0,
                    i18next.t("修改Toast弹出的顺序")
                  )
                ]
              }
            ]
          },
          UISelect(
            i18next.t("语言"),
            "setting-language",
            "zh-CN",
            [
              {
                value: "zh-CN",
                text: "中文"
              },
              {
                value: "en-US",
                text: "English"
              }
            ],
            (event, isSelectValue, isSelectText) => {
              log.info("改变语言:" + isSelectText);
              i18next.changeLanguage(isSelectValue);
            }
          )
        ]
      },
      {
        text: "",
        type: "forms",
        forms: [
          {
            text: i18next.t("账号/密码"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  UIInput(
                    i18next.t("账号"),
                    "user",
                    "",
                    void 0,
                    void 0,
                    i18next.t("请输入账号")
                  ),
                  UIInput(
                    i18next.t("密码"),
                    "pwd",
                    "",
                    void 0,
                    void 0,
                    i18next.t("请输入密码"),
                    false,
                    true
                  )
                ]
              },
              {
                text: "",
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("自动登录"),
                    "autoLogin",
                    true,
                    void 0,
                    i18next.t("自动登录当前保存的账号")
                  ),
                  UIButton(
                    i18next.t("清空账号/密码"),
                    void 0,
                    i18next.t("点击清空"),
                    void 0,
                    void 0,
                    false,
                    "default",
                    (event) => {
                      if (confirm(i18next.t("确定清空账号和密码?"))) {
                        PopsPanel.deleteValue("user");
                        PopsPanel.deleteValue("pwd");
                        Qmsg.success(i18next.t("已清空账号/密码"));
                        let $shadowRoot = event.target.getRootNode();
                        $shadowRoot.querySelector(
                          `li[data-key="user"] .pops-panel-input input`
                        ).value = "";
                        $shadowRoot.querySelector(
                          `li[data-key="pwd"] .pops-panel-input input`
                        ).value = "";
                      }
                    }
                  )
                ]
              }
            ]
          },
          {
            text: i18next.t("功能"),
            type: "deepMenu",
            forms: [
              {
                text: i18next.t("功能"),
                type: "forms",
                forms: [
                  UISelect(
                    i18next.t("固定当前语言"),
                    "language-selector-locale",
                    "",
                    function() {
                      let result = [
                        {
                          value: "",
                          text: i18next.t("无")
                        }
                      ];
                      document.querySelectorAll(
                        "select#language-selector-locale option"
                      ).forEach((element) => {
                        let value = element.getAttribute("value");
                        if (value === "help") {
                          return;
                        }
                        let text = (element.innerText || element.textContent).trim();
                        result.push({
                          value,
                          text
                        });
                      });
                      return result;
                    }()
                  ),
                  UISwitch(
                    i18next.t("修复图片宽度显示问题"),
                    "fixImageWidth",
                    true,
                    void 0,
                    i18next.t("修复图片在移动端宽度超出浏览器宽度问题")
                  ),
                  UISwitch(
                    i18next.t("优化图片浏览"),
                    "optimizeImageBrowsing",
                    true,
                    void 0,
                    i18next.t("使用Viewer浏览图片")
                  ),
                  UISwitch(
                    i18next.t("覆盖图床图片跳转"),
                    "overlayBedImageClickEvent",
                    true,
                    void 0,
                    i18next.t("配合上面的【优化图片浏览】更优雅浏览图片")
                  ),
                  UISwitch(
                    i18next.t("添加【操作面板】按钮"),
                    "scripts-addOperationPanelBtnWithNavigator",
                    true,
                    void 0,
                    i18next.t("在脚本列表页面时为顶部导航栏添加【操作面板】按钮")
                  ),
                  UISwitch(
                    i18next.t("给Markdown添加【复制】按钮"),
                    "addMarkdownCopyButton",
                    true,
                    void 0,
                    i18next.t(
                      "在Markdown内容右上角添加【复制】按钮,点击一键复制Markdown内容"
                    )
                  )
                ]
              },
              {
                text: i18next.t("检测页面加载"),
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("启用"),
                    "checkPage",
                    true,
                    void 0,
                    i18next.t(
                      "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面"
                    )
                  ),
                  UISelect(
                    i18next.t("检测间隔"),
                    "greasyfork-check-page-timeout",
                    5,
                    (() => {
                      let result = [];
                      for (let index = 0; index < 5; index++) {
                        result.push({
                          value: index + 1,
                          text: index + 1 + "s"
                        });
                      }
                      return result;
                    })(),
                    void 0,
                    i18next.t(
                      "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面"
                    )
                  )
                ]
              },
              {
                text: i18next.t("代码同步"),
                type: "forms",
                forms: [
                  UIButton(
                    i18next.t("源代码同步【脚本列表】"),
                    void 0,
                    i18next.t("一键同步"),
                    void 0,
                    void 0,
                    false,
                    "primary",
                    (event) => {
                      if (!GreasyforkRouter.isUserHome()) {
                        PopsPanel.setValue(
                          "goto_updateSettingsAndSynchronize_scriptList",
                          true
                        );
                        if (GreasyforkMenu.getUserLinkElement()) {
                          Qmsg.success(i18next.t("前往用户主页"));
                          window.location.href = GreasyforkMenu.getUserLinkElement().href;
                        } else {
                          Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
                        }
                        return;
                      }
                      let scriptUrlList = [];
                      document.querySelectorAll(
                        "#user-script-list-section li a.script-link"
                      ).forEach((item) => {
                        scriptUrlList = scriptUrlList.concat(
                          GreasyforkApi.getAdminUrl(item.href)
                        );
                      });
                      GreasyforkMenu.updateScript(scriptUrlList);
                    }
                  ),
                  UIButton(
                    i18next.t("源代码同步【未上架的脚本】"),
                    void 0,
                    i18next.t("一键同步"),
                    void 0,
                    void 0,
                    false,
                    "primary",
                    (event) => {
                      if (!GreasyforkRouter.isUserHome()) {
                        PopsPanel.setValue(
                          "goto_updateSettingsAndSynchronize_unlistedScriptList",
                          true
                        );
                        if (GreasyforkMenu.getUserLinkElement()) {
                          Qmsg.success(i18next.t("前往用户主页"));
                          window.location.href = GreasyforkMenu.getUserLinkElement().href;
                        } else {
                          Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
                        }
                        return;
                      }
                      let scriptUrlList = [];
                      document.querySelectorAll(
                        "#user-unlisted-script-list li a.script-link"
                      ).forEach((item) => {
                        scriptUrlList = scriptUrlList.concat(
                          GreasyforkApi.getAdminUrl(item.href)
                        );
                      });
                      GreasyforkMenu.updateScript(scriptUrlList);
                    }
                  ),
                  UIButton(
                    i18next.t("源代码同步【库】"),
                    void 0,
                    i18next.t("一键同步"),
                    void 0,
                    void 0,
                    false,
                    "primary",
                    (event) => {
                      if (!GreasyforkRouter.isUserHome()) {
                        PopsPanel.setValue(
                          "goto_updateSettingsAndSynchronize_libraryScriptList",
                          true
                        );
                        if (GreasyforkMenu.getUserLinkElement()) {
                          Qmsg.success(i18next.t("前往用户主页"));
                          window.location.href = GreasyforkMenu.getUserLinkElement().href;
                        } else {
                          Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
                        }
                        return;
                      }
                      let scriptUrlList = [];
                      document.querySelectorAll(
                        "#user-library-script-list li a.script-link"
                      ).forEach((item) => {
                        scriptUrlList = scriptUrlList.concat(
                          GreasyforkApi.getAdminUrl(item.href)
                        );
                      });
                      GreasyforkMenu.updateScript(scriptUrlList);
                    }
                  )
                ]
              }
            ]
          },
          {
            text: i18next.t("美化"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("美化页面元素"),
                    "beautifyPage",
                    true,
                    void 0,
                    i18next.t("如button、input、textarea")
                  ),
                  UISwitch(
                    i18next.t("美化上传图片按钮"),
                    "beautifyUploadImage",
                    true,
                    void 0,
                    i18next.t("放大上传区域")
                  ),
                  UISwitch(
                    i18next.t("美化顶部导航栏"),
                    "beautifyTopNavigationBar",
                    true,
                    void 0,
                    i18next.t("可能会跟Greasyfork Beautify脚本有冲突")
                  ),
                  UISwitch(
                    i18next.t("美化Greasyfork Beautify脚本"),
                    "beautifyGreasyforkBeautify",
                    true,
                    void 0,
                    i18next.t(
                      '需安装Greasyfork Beautify脚本,<a href="https://greasyfork.org/zh-CN/scripts/446849-greasyfork-beautify" target="_blank">🖐点我安装</a>'
                    )
                  )
                ]
              }
            ]
          }
        ]
      }
    ]
  };
  const GreasyforkScriptsCode = {
    init() {
      PopsPanel.execMenuOnce("code-repairCodeLineNumber", () => {
        this.repairCodeLineNumber();
      });
    },
    /**
     * 修复代码的行号显示不够问题
     * 超过1w行不会高亮代码
     */
    repairCodeLineNumber() {
      log.info("修复代码的行号显示不够问题");
      PopsPanel.execMenuOnce("beautifyGreasyforkBeautify", () => {
        _GM_addStyle(
          /*css*/
          `
				.code-container pre code .marker{
					padding-left: 6px;
				}	
				`
        );
      });
      utils.waitNode(
        "#script-content div.code-container pre.prettyprint ol"
      ).then(($prettyPrintOL) => {
        if ($prettyPrintOL.childElementCount >= 1e3) {
          log.success(
            `当前代码行数${$prettyPrintOL.childElementCount}行,超过1000行,优化行号显示问题`
          );
          _GM_addStyle(
            /*css*/
            `
                    pre.prettyprint{
                        padding-left: 26px;
                    }
					`
          );
        }
      });
    }
  };
  const beautifyVersionsPageCSS = 'ul.history_versions,\r\nul.history_versions li {\r\n	width: 100%;\r\n}\r\nul.history_versions li {\r\n	display: flex;\r\n	flex-direction: column;\r\n	margin: 25px 0px;\r\n}\r\n.diff-controls input[type="radio"]:nth-child(2) {\r\n	margin-left: 5px;\r\n}\r\n.flex-align-item-center {\r\n	display: flex;\r\n	align-items: center;\r\n}\r\n.script-tag {\r\n	margin-bottom: 8px;\r\n}\r\n.script-tag-version a {\r\n	color: #656d76;\r\n	fill: #656d76;\r\n	text-decoration: none;\r\n	width: fit-content;\r\n	width: -moz-fit-content;\r\n}\r\n.script-tag-version a:hover svg {\r\n	color: #00a3f5;\r\n	fill: #00a3f5;\r\n}\r\n.script-tag-version a > span {\r\n	margin-left: 0.25rem;\r\n}\r\n.script-note-box-body {\r\n	border-radius: 0.375rem;\r\n	border-style: solid;\r\n	border-width: max(1px, 0.0625rem);\r\n	border-color: #d0d7de;\r\n	color: #1f2328;\r\n	padding: 16px;\r\n	overflow-wrap: anywhere;\r\n}\r\n.script-note-box-body p {\r\n	margin-bottom: unset;\r\n}\r\n';
  const CommonUtils = {
    /**
     * 添加屏蔽CSS
     * @param args
     * @example
     * addBlockCSS("")
     * addBlockCSS("","")
     * addBlockCSS(["",""])
     */
    addBlockCSS(...args) {
      let selectorList = [];
      if (args.length === 0) {
        return;
      }
      if (args.length === 1 && typeof args[0] === "string" && args[0].trim() === "") {
        return;
      }
      args.forEach((selector) => {
        if (Array.isArray(selector)) {
          selectorList = selectorList.concat(selector);
        } else {
          selectorList.push(selector);
        }
      });
      return addStyle(`${selectorList.join(",\n")}{display: none !important;}`);
    },
    /**
     * 设置GM_getResourceText的style内容
     * @param resourceMapData 资源数据
     */
    setGMResourceCSS(resourceMapData) {
      let cssText = typeof _GM_getResourceText === "function" ? _GM_getResourceText(resourceMapData.keyName) : "";
      if (typeof cssText === "string" && cssText) {
        addStyle(cssText);
      } else {
        CommonUtils.addLinkNode(resourceMapData.url);
      }
    },
    /**
     * 添加<link>标签
     * @param url
     */
    async addLinkNode(url) {
      let $link = document.createElement("link");
      $link.rel = "stylesheet";
      $link.type = "text/css";
      $link.href = url;
      domUtils.ready(() => {
        document.head.appendChild($link);
      });
      return $link;
    },
    /**
     * 将url修复,例如只有search的链接/sss/xxx?sss=xxxx
     * @param url 需要修复的链接
     */
    fixUrl(url) {
      url = url.trim();
      if (url.match(/^http(s|):\/\//i)) {
        return url;
      } else {
        if (!url.startsWith("/")) {
          url += "/";
        }
        url = window.location.origin + url;
        return url;
      }
    }
  };
  const GreasyforkUrlUtils = {
    /**
     * 获取脚本安装的链接
     * @param scriptId
     * @param scriptVersion
     * @param scriptName
     * @returns
     */
    getInstallUrl(scriptId, scriptVersion, scriptName) {
      if (utils.isNotNull(scriptName)) {
        scriptName = "/" + scriptName;
      } else {
        scriptName = "";
      }
      return `https://update.greasyfork.org/scripts/${scriptId}/${scriptVersion}${scriptName}.user.js`;
    },
    /**
     * 获取脚本的代码页面链接
     * @param scriptId
     * @param scriptVersion
     * @returns
     */
    getCodeUrl(scriptId, scriptVersion) {
      if (utils.isNull(scriptVersion)) {
        scriptVersion = "";
      }
      return `https://greasyfork.org/scripts/${scriptId}/code?version=${scriptVersion}`;
    },
    /**
     * 获取脚本的信息
     * @param scriptId 脚本id
     */
    getScriptInfoUrl(scriptId) {
      return `https://greasyfork.org/scripts/${scriptId}.json`;
    }
  };
  const GreasyforkVersions = {
    init() {
      PopsPanel.execMenuOnce("beautifyHistoryVersionPage", () => {
        return this.beautifyHistoryVersionPage();
      });
      PopsPanel.execMenuOnce("scripts-versions-addExtraTagButton", () => {
        this.addExtraTagButton();
      });
    },
    /**
     * 美化 历史版本 页面
     */
    beautifyHistoryVersionPage() {
      log.info("美化 历史版本 页面");
      let result = [];
      result.push(_GM_addStyle(beautifyVersionsPageCSS));
      result.push(
        CommonUtils.addBlockCSS(
          ".version-number",
          ".version-date",
          ".version-changelog"
        )
      );
      domUtils.ready(function() {
        let $historyVersion = document.querySelector(
          "ul.history_versions"
        );
        if (!$historyVersion) {
          Qmsg.error(i18next.t("未找到history_versions元素列表"));
          return;
        }
        Array.from($historyVersion.children).forEach((liElement) => {
          var _a2, _b;
          let versionUrl = liElement.querySelector(".version-number a").href;
          let versionNumber = liElement.querySelector(
            ".version-number a"
          ).innerText;
          let versionDate = (_a2 = liElement.querySelector(".version-date")) == null ? void 0 : _a2.getAttribute("datetime");
          let updateNote = ((_b = liElement.querySelector(".version-changelog")) == null ? void 0 : _b.innerHTML) || "";
          let versionDateElement = domUtils.createElement("span", {
            className: "script-version-date",
            innerHTML: utils.formatTime(
              versionDate,
              i18next.t("yyyy年MM月dd日 HH:mm:ss")
            )
          });
          let tagElement = domUtils.createElement("div", {
            className: "script-tag",
            innerHTML: (
              /*html*/
              `
                    <div class="script-tag-version">
                        <a href="${versionUrl}" class="flex-align-item-center">
                        <svg aria-label="Tag" role="img" height="16" viewBox="0 0 16 16" version="1.1" width="16">
                            <path d="M1 7.775V2.75C1 1.784 1.784 1 2.75 1h5.025c.464 0 .91.184 1.238.513l6.25 6.25a1.75 1.75 0 0 1 0 2.474l-5.026 5.026a1.75 1.75 0 0 1-2.474 0l-6.25-6.25A1.752 1.752 0 0 1 1 7.775Zm1.5 0c0 .066.026.13.073.177l6.25 6.25a.25.25 0 0 0 .354 0l5.025-5.025a.25.25 0 0 0 0-.354l-6.25-6.25a.25.25 0 0 0-.177-.073H2.75a.25.25 0 0 0-.25.25ZM6 5a1 1 0 1 1 0 2 1 1 0 0 1 0-2Z"></path>
                        </svg>
                        <span>${versionNumber}</span>
                        </a>
                    </div>`
            )
          });
          let boxBodyElement = domUtils.createElement("div", {
            className: "script-note-box-body",
            innerHTML: updateNote
          });
          liElement.appendChild(versionDateElement);
          liElement.appendChild(tagElement);
          liElement.appendChild(boxBodyElement);
        });
      });
      return result;
    },
    /**
     * 添加额外的标签按钮
     */
    addExtraTagButton() {
      log.info("添加额外的标签按钮");
      domUtils.ready(() => {
        document.querySelectorAll(".script-tag-version").forEach(($tagVersion) => {
          var _a2, _b;
          let $anchor = $tagVersion.querySelector("a");
          if (!$anchor) {
            return;
          }
          let urlObj = new URL($anchor.href);
          let scriptId = (_a2 = urlObj.pathname.match(/\/scripts\/([\d]+)/)) == null ? void 0 : _a2[1];
          let scriptVersion = urlObj.searchParams.get("version");
          let scriptName = (_b = urlObj.pathname.match(/\/scripts\/[\d]+-(.+)/)) == null ? void 0 : _b[1];
          let installUrl = GreasyforkUrlUtils.getInstallUrl(
            scriptId,
            scriptVersion,
            scriptName
          );
          let codeUrl = GreasyforkUrlUtils.getCodeUrl(scriptId, scriptVersion);
          let $buttonTag = domUtils.createElement("div", {
            className: "scripts-tag-install",
            innerHTML: (
              /*html*/
              `
						<a class="script-btn-install install-link" data-install-format="js" target="_blank" href="${installUrl}">${i18next.t(
              "安装此脚本"
            )}</a>
						<a class="script-btn-see-code" target="_blank" href="${codeUrl}">${i18next.t(
              "查看代码"
            )}</a>
						`
            )
          });
          domUtils.after($tagVersion, $buttonTag);
        });
      });
    }
  };
  let userCollection = [];
  const GreasyforkScriptsCollectEvent = async function(scriptId) {
    log.info("当前脚本id:" + scriptId);
    if (!GreasyforkMenu.isLogin) {
      log.error("请先登录账号");
      Qmsg.error(i18next.t("请先登录账号"));
      return;
    }
    let userId = GreasyforkApi.getUserId(
      GreasyforkMenu.getUserLinkElement().href
    );
    if (userId == null) {
      log.error("获取用户id失败");
      Qmsg.error(i18next.t("获取用户id失败"));
      return;
    }
    if (!userCollection.length) {
      let loading = Qmsg.loading(i18next.t("获取收藏夹中..."));
      userCollection = await GreasyforkApi.getUserCollection(userId) || [];
      loading.close();
      if (!userCollection.length) {
        return;
      }
    }
    let alertHTML = "";
    userCollection.forEach((userCollectInfo) => {
      alertHTML += /*html*/
      `
		<li class="user-collect-item" data-id="${userCollectInfo.id}" data-name="${userCollectInfo.name}">
			<div class="user-collect-name">${userCollectInfo.name}</div>
			<div class="user-collect-btn-container">
			<div class="pops-panel-button collect-add-script-id">
				<button type="primary" data-icon="" data-righticon="">
				<span>${i18next.t("添加")}</span>
				</button>
			</div>
			<div class="pops-panel-button collect-delete-script-id">
				<button type="danger" data-icon="" data-righticon="">
				<span>${i18next.t("刪除")}</span>
				</button>
			</div>
			</div>
		</li>
		  `;
    });
    let collectionDialog = __pops.alert({
      title: {
        text: i18next.t("收藏集"),
        position: "center"
      },
      content: {
        html: true,
        text: (
          /*html*/
          `<ul>${alertHTML}</ul>`
        )
      },
      mask: {
        enable: true,
        clickEvent: {
          toClose: true
        }
      },
      btn: {
        ok: {
          enable: false
        }
      },
      width: __pops.isPhone() ? "92vw" : "500px",
      height: "auto",
      drag: true,
      only: true,
      style: (
        /*css*/
        `
		.pops{
			--content-max-height: 400px;
			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));
		}
		.user-collect-item{
			-webkit-user-select: none;
			user-select: none;
			padding: 5px 10px;
			display: flex;
			align-items: center;
			justify-content: space-between;
			border-bottom: 1px dotted #c9c9c9;
		}
		.user-collect-name{

		}
		.user-collect-item:hover{
			
		}
		.user-collect-btn-container{
			margin-left: 10px;
			display: flex;
		}
		`
      )
    });
    domUtils.on(
      collectionDialog.$shadowRoot,
      "click",
      ".collect-add-script-id",
      async function(event) {
        let $userCollectItem = event.target.closest(
          ".user-collect-item"
        );
        let setsId = $userCollectItem.dataset.id;
        $userCollectItem.dataset.name;
        let loading = Qmsg.loading(i18next.t("添加中..."));
        let formData = await GreasyforkApi.getUserCollectionInfo(userId, setsId);
        if (!formData) {
          loading.close();
          return;
        }
        let editForm = utils.cloneFormData(formData);
        let saveEditForm = utils.cloneFormData(formData);
        let isCollect = false;
        for (const [key, value] of formData.entries()) {
          if (key === "scripts-included[]" && String(value).trim() === String(scriptId).trim()) {
            isCollect = true;
            break;
          } else {
            saveEditForm.append(key, value);
            editForm.append(key, value);
          }
        }
        if (isCollect) {
          Qmsg.warning(i18next.t("该脚本已经在该收藏集中"));
          loading.close();
          return;
        }
        editForm.set("add-script", scriptId.toString());
        editForm.set("script-action", "i");
        saveEditForm.append("scripts-included[]", scriptId.toString());
        saveEditForm.set("save", "1");
        let addFormDataSearchParams = new URLSearchParams(editForm);
        let saveFormDataSearchParams = new URLSearchParams(saveEditForm);
        let addData = Array.from(addFormDataSearchParams).map(
          // @ts-ignore
          ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
        ).join("&");
        let saveData = Array.from(saveFormDataSearchParams).map(
          // @ts-ignore
          ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
        ).join("&");
        log.info(["添加的数据", addData]);
        log.info(["保存的数据", saveData]);
        let addResult = await GreasyforkApi.updateUserSetsInfo(
          userId,
          setsId,
          addData
        );
        if (!addResult) {
          loading.close();
          return;
        }
        let changeScriptSet = addResult.querySelector(".change-script-set");
        if (!changeScriptSet) {
          Qmsg.error(
            i18next.t("添加失败,{{selector}}元素不存在", {
              selector: ".change-script-set"
            })
          );
          loading.close();
          return;
        }
        let section = changeScriptSet.querySelector("section");
        if (!section) {
          Qmsg.error(
            i18next.t("添加失败,{{selector}}元素不存在", {
              selector: "section"
            })
          );
          loading.close();
          return;
        }
        let alertElement = section.querySelector(".alert");
        if (alertElement) {
          __pops.alert({
            title: {
              text: i18next.t("添加失败"),
              position: "center"
            },
            content: {
              text: alertElement.innerHTML,
              html: true
            },
            mask: {
              enable: true,
              clickEvent: {
                toClose: true
              }
            },
            style: (
              /*css*/
              `
					.pops-alert-content{
						font-style: italic;
						background-color: #ffc;
						border: none;
						border-left: 6px solid #FFEB3B;
						padding: .5em;
					}
					`
            ),
            drag: true,
            dragLimit: true,
            width: __pops.isPhone() ? "88vw" : "400px",
            height: __pops.isPhone() ? "50vh" : "300px"
          });
        } else {
          await GreasyforkApi.updateUserSetsInfo(userId, setsId, saveData);
          Qmsg.success(i18next.t("添加成功"));
        }
        loading.close();
      }
    );
    domUtils.on(
      collectionDialog.$shadowRoot,
      "click",
      ".collect-delete-script-id",
      async function(event) {
        let $collectItem = event.target.closest(
          ".user-collect-item"
        );
        let setsId = $collectItem.dataset.id;
        $collectItem.dataset.name;
        let loading = Qmsg.loading(i18next.t("删除中..."));
        let formData = await GreasyforkApi.getUserCollectionInfo(userId, setsId);
        if (!formData) {
          loading.close();
          return;
        }
        let editForm = new FormData();
        let saveEditForm = new FormData();
        let isCollect = false;
        for (const [key, value] of formData.entries()) {
          if (String(key).trim() === "scripts-included[]" && String(value).trim() === String(scriptId).trim()) {
            isCollect = true;
            continue;
          }
          saveEditForm.append(key, value);
          editForm.append(key, value);
        }
        if (!isCollect) {
          Qmsg.warning(
            i18next.t("该收藏集未包含:{{scriptId}}", {
              scriptId
            })
          );
          loading.close();
          return;
        }
        editForm.set("remove-scripts-included[]", scriptId.toString());
        editForm.set("remove-selected-scripts", "i");
        editForm.delete("script-action");
        saveEditForm.set("save", "1");
        let deleteFormDataSearchParams = new URLSearchParams(editForm);
        let saveFormDataSearchParams = new URLSearchParams(saveEditForm);
        let removeData = Array.from(deleteFormDataSearchParams).map(
          // @ts-ignore
          ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
        ).join("&");
        let saveData = Array.from(saveFormDataSearchParams).map(
          // @ts-ignore
          ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
        ).join("&");
        log.info(["删除的数据", removeData]);
        log.info(["保存的数据", saveData]);
        let removeResult = await GreasyforkApi.updateUserSetsInfo(
          userId,
          setsId,
          removeData
        );
        if (!removeResult) {
          loading.close();
          return;
        }
        await GreasyforkApi.updateUserSetsInfo(userId, setsId, saveData);
        Qmsg.success(i18next.t("删除成功"));
        loading.close();
      }
    );
  };
  const GreasyforkScripts = {
    init() {
      if (GreasyforkRouter.isCode()) {
        GreasyforkScriptsCode.init();
      } else if (GreasyforkRouter.isVersion()) {
        GreasyforkVersions.init();
      }
      if (GreasyforkRouter.isCodeStrict()) {
        PopsPanel.execMenuOnce("fullScreenOptimization", () => {
          this.fullScreenOptimization();
        });
        PopsPanel.execMenuOnce("addCopyCodeButton", () => {
          this.addCopyCodeButton();
        });
      }
      PopsPanel.execMenuOnce("addCollectionButton", () => {
        this.addCollectionButton();
      });
      PopsPanel.execMenuOnce("addFindReferenceButton", () => {
        this.setFindCodeSearchBtn();
      });
      PopsPanel.execMenuOnce("scriptHomepageAddedTodaySUpdate", () => {
        this.scriptHomepageAddedTodaySUpdate();
      });
    },
    /**
     * 添加【收藏】按钮
     */
    addCollectionButton() {
      log.info("添加收藏按钮");
      utils.waitNode("ul#script-links li.current span").then(() => {
        let $collectBtn = domUtils.createElement("li", {
          innerHTML: `
					<a href="javascript:;">
						<span>${i18next.t("收藏")}</span>
					</a>`
        });
        domUtils.append(
          document.querySelector("ul#script-links"),
          $collectBtn
        );
        domUtils.on($collectBtn, "click", () => {
          let scriptIdMatch = window.location.pathname.match(/scripts\/([\d]+)/i);
          if (!scriptIdMatch) {
            log.error([scriptIdMatch, window.location.pathname]);
            Qmsg.error(i18next.t("获取脚本id失败"));
            return;
          }
          let scriptId = scriptIdMatch[scriptIdMatch.length - 1];
          GreasyforkScriptsCollectEvent(scriptId);
        });
      });
    },
    /**
     * F11全屏,F键代码全屏
     */
    fullScreenOptimization() {
      log.info("F11全屏,F键代码全屏");
      _GM_addStyle(
        /*css*/
        `
        .code-wide-screen{
          position: absolute;
          top: 0;
          left: 0;
          right: 0;
          bottom: 0;
          margin: 0;
          padding: 0;
          width: 100%;
          height: 100%;
          min-width: 100%;
          min-height: 100%;
          max-width: 100%;
          max-height: 100%;
          z-index: 10000;
        }
        `
      );
      let isFullScreen = false;
      domUtils.keydown(
        _unsafeWindow,
        function(event) {
          if (event.key.toLowerCase() === "f") {
            let codeElement = document.querySelector(
              "#script-content div.code-container code"
            );
            if (event.altKey && event.shiftKey) {
              utils.preventEvent(event);
              if (codeElement.classList.contains("code-wide-screen")) {
                codeElement.classList.remove("code-wide-screen");
              } else {
                codeElement.classList.add("code-wide-screen");
              }
            } else if (!event.altKey && !event.ctrlKey && !event.shiftKey && !event.metaKey) {
              utils.preventEvent(event);
              if (isFullScreen) {
                utils.exitFullScreen(codeElement);
                isFullScreen = false;
              } else {
                utils.enterFullScreen(codeElement);
                isFullScreen = true;
              }
            }
          }
        },
        {
          capture: true
        }
      );
    },
    /**
     * 设置代码搜索按钮(对于库)
     */
    setFindCodeSearchBtn() {
      log.info("设置代码搜索按钮(对于库)");
      utils.waitNode("ul#script-links li.current span").then(() => {
        let searchBtn = domUtils.createElement("li", {
          innerHTML: `
					<a href="javascript:;">
						<span>${i18next.t("寻找引用")}</span>
					</a>`
        });
        domUtils.append(
          document.querySelector(
            "ul#script-links"
          ),
          searchBtn
        );
        domUtils.on(searchBtn, "click", async function() {
          let scriptIdMatch = window.location.pathname.match(/scripts\/([\d]+)/i);
          if (!scriptIdMatch) {
            log.error([scriptIdMatch, window.location.pathname]);
            Qmsg.error(i18next.t("获取脚本id失败"));
            return;
          }
          let scriptId = scriptIdMatch[scriptIdMatch.length - 1];
          window.location.href = GreasyforkApi.getCodeSearchUrl(
            `greasyfork.org/scripts/${scriptId}`
          );
        });
      });
    },
    /**
     * 脚本首页新增【今日检查】
     */
    async scriptHomepageAddedTodaySUpdate() {
      if (!document.querySelector("#install-area")) {
        return;
      }
      log.info("脚本首页新增【今日检查】");
      let scriptStatsJSONInfo = await GreasyforkApi.getScriptStats(
        GreasyforkApi.getScriptId()
      );
      if (!scriptStatsJSONInfo) {
        return;
      }
      let scriptStatsJSON = utils.toJSON(scriptStatsJSONInfo.responseText);
      log.info(["统计信息", scriptStatsJSON]);
      let todayStatsJSON = scriptStatsJSON[utils.formatTime(void 0, "yyyy-MM-dd")];
      if (!todayStatsJSON) {
        log.error("今日份的统计信息不存在");
        return;
      }
      let update_checks = todayStatsJSON["update_checks"];
      log.info(["今日统计信息", todayStatsJSON]);
      domUtils.after(
        "dd.script-show-daily-installs",
        domUtils.createElement("dt", {
          className: "script-show-daily-update_checks",
          innerHTML: `<span>${i18next.t("今日检查")}</span>`
        })
      );
      domUtils.after(
        "dt.script-show-daily-update_checks",
        domUtils.createElement("dd", {
          className: "script-show-daily-update_checks",
          innerHTML: "<span>" + update_checks + "</span>"
        })
      );
    },
    /**
     * 添加复制代码按钮
     */
    addCopyCodeButton() {
      log.info("添加复制代码按钮");
      utils.waitNode("div#script-content div.code-container").then(($codeContainer) => {
        let copyButton = domUtils.createElement(
          "button",
          {
            textContent: i18next.t("复制代码")
          },
          {
            style: "margin-bottom: 1em;"
          }
        );
        domUtils.on(copyButton, "click", async function() {
          let loading = Qmsg.loading(i18next.t("加载文件中..."));
          let getResp = await httpx.get(
            `https://greasyfork.org/zh-CN/scripts/${GreasyforkApi.getScriptId()}.json`,
            {
              fetch: true,
              responseType: "json"
            }
          );
          if (!getResp.status) {
            loading.close();
            return;
          }
          let respJSON = utils.toJSON(getResp.data.responseText);
          let code_url = respJSON["code_url"];
          log.success(["代码地址:", code_url]);
          let scriptJS = await httpx.get(code_url);
          if (!scriptJS.status) {
            loading.close();
            return;
          }
          loading.close();
          utils.setClip(scriptJS.data.responseText);
          Qmsg.success(i18next.t("复制成功"));
        });
        domUtils.before($codeContainer, copyButton);
      });
    }
  };
  const beautifyCenterContentCSS = '.sidebarred-main-content {\r\n	max-width: unset;\r\n	flex: unset;\r\n}\r\nol.script-list {\r\n	display: flex;\r\n	flex-wrap: wrap;\r\n	border: none;\r\n	gap: 20px;\r\n	background: transparent;\r\n	box-shadow: none;\r\n}\r\nol.script-list .script-description {\r\n	overflow-wrap: anywhere;\r\n}\r\nol.script-list li {\r\n	border: 1px solid rgb(221, 221, 221);\r\n	border-radius: 5px;\r\n	flex: 1 1 45%;\r\n	box-shadow: rgb(221, 221, 221) 0px 0px 5px 2px;\r\n}\r\n/* 收藏按钮 */\r\n.script-collect-btn {\r\n	color: #ffffff;\r\n	border-color: #409eff;\r\n	background-color: #409eff;\r\n}\r\n/* 评分按钮 */\r\n.script-list-rating-score[data-position="right"] {\r\n	display: inline-block;\r\n	min-width: 1em;\r\n	text-align: center;\r\n	padding: 0 0.25em;\r\n	border: 1px solid #dddddd;\r\n	border-radius: 10px;\r\n	width: fit-content;\r\n}\r\n\r\n/* 加载圆圈动画 */\r\n.install-link.lum-lightbox-loader {\r\n	position: relative;\r\n	min-width: 4rem;\r\n	min-height: 1rem;\r\n}\r\n.install-link.lum-lightbox-loader::before {\r\n	margin-left: 1rem;\r\n}\r\n.install-link.lum-lightbox-loader::after {\r\n	margin-right: 1rem;\r\n}\r\n.install-link.lum-lightbox-loader::before,\r\n.install-link.lum-lightbox-loader::after {\r\n	width: 1em;\r\n	height: 1em;\r\n	margin-top: -0.5em;\r\n	border-radius: 1em;\r\n	background: hsla(0, 0%, 100%, 0.5);\r\n}\r\n';
  const GreasyforkCheckVersion = {
    /** 获取 TamperMonkey 暴露在window下的函数 */
    getTampermonkey: () => {
      var _a2;
      return (_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Tampermonkey;
    },
    /** 获取 Violentmonkey 暴露在window下的函数 */
    getViolentmonkey: () => {
      var _a2;
      return (_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Violentmonkey;
    },
    /** 获取 ScriptCat 暴露在window下的函数 */
    getScriptCat: () => {
      var _a2;
      return (_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Scriptcat;
    },
    /**
     * 获取脚本容器启用状态
     */
    getScriptContainerStatus() {
      var _a2, _b, _c;
      let containerStatus = {
        Tampermonkey: false,
        Violentmonkey: false,
        ScriptCat: false
      };
      if ((_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Tampermonkey) {
        containerStatus.Tampermonkey = true;
      }
      if ((_b = _unsafeWindow.external) == null ? void 0 : _b.Violentmonkey) {
        containerStatus.Violentmonkey = true;
      }
      if ((_c = _unsafeWindow.external) == null ? void 0 : _c.Scriptcat) {
        containerStatus.ScriptCat = true;
      }
      return containerStatus;
    },
    /**
     * 获取脚本安装的版本号
     * @param name 脚本名
     * @param namespace 脚本命名空间
     */
    getInstalledVersion(name, namespace) {
      return new Promise((resolve, reject) => {
        const tm = this.getTampermonkey();
        if (tm) {
          tm.isInstalled(name, namespace, function(data) {
            if (data.installed) {
              resolve(data.version);
            } else {
              resolve(null);
            }
          });
          return;
        }
        const vm = this.getViolentmonkey();
        if (vm) {
          vm.isInstalled(name, namespace).then(resolve);
          return;
        }
        const scriptCat = this.getScriptCat();
        if (scriptCat) {
          scriptCat.isInstalled(name, namespace, function(data) {
            if (data.installed) {
              resolve(data.version);
            } else {
              resolve(null);
            }
          });
          return;
        }
        reject(new TypeError("获取脚本容器暴露的external信息失败"));
      });
    },
    /**
     * https://developer.mozilla.org/en/docs/Toolkit_version_format
     *
     * 比较版本号
     * @param a 版本号
     * @param b 版本号
     * @returns
     * + -1 该版本号低
     * + 0 该版本号和比较的版本号相同
     * + 1 该版本号高
     */
    compareVersions(a, b) {
      if (a === b) {
        return 0;
      }
      const aParts = a.split(".");
      const bParts = b.split(".");
      for (let i = 0; i < aParts.length; i++) {
        const result = this.compareVersionPart(aParts[i], bParts[i]);
        if (result !== 0) {
          return result;
        }
      }
      return 0;
    },
    compareVersionPart(partA, partB) {
      const partAParts = this.parseVersionPart(partA);
      const partBParts = this.parseVersionPart(partB);
      for (let i = 0; i < partAParts.length; i++) {
        if (partAParts[i].length > 0 && partBParts[i].length === 0) {
          return -1;
        }
        if (partAParts[i].length === 0 && partBParts[i].length > 0) {
          return 1;
        }
        if (partAParts[i] > partBParts[i]) {
          return 1;
        }
        if (partAParts[i] < partBParts[i]) {
          return -1;
        }
      }
      return 0;
    },
    // It goes number, string, number, string. If it doesn't exist, then
    // 0 for numbers, empty string for strings.
    parseVersionPart(part) {
      if (!part) {
        return [0, "", 0, ""];
      }
      const partParts = /([0-9]*)([^0-9]*)([0-9]*)([^0-9]*)/.exec(part);
      return [
        partParts[1] ? parseInt(partParts[1]) : 0,
        partParts[2],
        partParts[3] ? parseInt(partParts[3]) : 0,
        partParts[4]
      ];
    },
    /**
     *
     * @param installButton 安装按钮
     * 必须属性
     * + data-update-label 按钮升级的文字
     * + data-downgrade-label 按钮降级的文字
     * + data-reinstall-label 按钮重装的文字
     * @param installedVersion 安装版本
     * @param version 版本号
     * @returns
     */
    handleInstallResult(installButton, installedVersion, version) {
      if (installedVersion == null) {
        return;
      }
      installButton.removeAttribute("data-ping-url");
      switch (this.compareVersions(installedVersion, version)) {
        case -1:
          installButton.textContent = installButton.getAttribute("data-update-label");
          break;
        case 1:
          installButton.textContent = installButton.getAttribute(
            "data-downgrade-label"
          );
          break;
        case 0:
          installButton.textContent = installButton.getAttribute(
            "data-reinstall-label"
          );
          break;
      }
    },
    /**
     * 检测js脚本的更新
     * @param installButton 安装按钮
     * 必须属性
     * + data-script-name 脚本名
     * + data-script-namespace 脚本命名空间
     * + data-script-version 脚本当前版本号
     * @param retry 重试
     */
    async checkForUpdatesJS(installButton, retry) {
      const name = installButton.getAttribute("data-script-name");
      const namespace = installButton.getAttribute("data-script-namespace");
      const version = installButton.getAttribute("data-script-version");
      try {
        let installedVersion = await this.getInstalledVersion(name, namespace);
        if (installedVersion == null) {
          return false;
        }
        this.handleInstallResult(installButton, installedVersion, version);
        return true;
      } catch (error) {
        if (retry) {
          await utils.sleep(1e3);
          try {
            return await this.checkForUpdatesJS(installButton, false);
          } catch (error2) {
          }
        }
        return false;
      }
    },
    /**
     * 检测css脚本的更新
     * @param installButton 安装按钮
     * 必须属性
     * + data-script-name 脚本名
     * + data-script-namespace 脚本命名空间
     */
    checkForUpdatesCSS(installButton) {
      const name = installButton.getAttribute("data-script-name");
      const namespace = installButton.getAttribute("data-script-namespace");
      postMessage(
        { type: "style-version-query", name, namespace, url: location.href },
        location.origin
      );
    }
  };
  const parseScriptListInfo = ($scriptList) => {
    let dataset = $scriptList.dataset;
    const info = {
      scriptId: parseInt(dataset.scriptId),
      scriptName: dataset.scriptName,
      scriptAuthors: [],
      scriptDailyInstalls: parseInt(dataset.scriptDailyInstalls),
      scriptTotalInstalls: parseInt(dataset.scriptTotalInstalls),
      scriptRatingScore: parseFloat(dataset.scriptRatingScore),
      scriptCreatedDate: new Date(dataset.scriptCreatedDate),
      scriptUpdatedDate: new Date(dataset.scriptUpdatedDate),
      scriptType: dataset.scriptType,
      scriptVersion: dataset.scriptVersion,
      sensitive: dataset.sensitive === "true",
      scriptLanguage: dataset.scriptLanguage,
      cssAvailableAsJs: dataset.cssAvailableAsJs === "true",
      codeUrl: dataset.codeUrl,
      scriptDescription: dataset.scriptDescription,
      scriptAuthorId: parseInt(dataset.scriptAuthorId),
      scriptAuthorName: dataset.scriptAuthorName
    };
    let scriptAuthorsObj = utils.toJSON(dataset.scriptAuthors);
    Object.keys(scriptAuthorsObj).forEach((authorId) => {
      let authorName = scriptAuthorsObj[authorId];
      info.scriptAuthors.push({
        authorId: parseInt(authorId),
        authorName
      });
    });
    if ((info.scriptAuthorName == null || isNaN(info.scriptAuthorId)) && info.scriptAuthors.length) {
      info.scriptAuthorName = info.scriptAuthors[0].authorName;
      info.scriptAuthorId = info.scriptAuthors[0].authorId;
    }
    if (info.scriptDescription == null) {
      let $description = $scriptList.querySelector(".script-description") || $scriptList.querySelector(".description");
      if ($description) {
        info.scriptDescription = $description.innerText || $description.textContent;
      }
    }
    return info;
  };
  const GreasyforkScriptsList = {
    init() {
      PopsPanel.execMenuOnce("gf-scripts-filter-enable", () => {
        GreasyforkScriptsFilter.init();
      });
      PopsPanel.execMenuOnce("beautifyCenterContent", () => {
        return this.beautifyCenterContent();
      });
    },
    /**
     * 美化脚本列表
     */
    beautifyCenterContent() {
      log.info("美化脚本列表");
      let result = [];
      result.push(_GM_addStyle(beautifyCenterContentCSS));
      DOMUtils.ready(async () => {
        let allScriptsList = GreasyforkScriptsFilter.getElementList();
        allScriptsList.forEach(($scriptList) => {
          if ($scriptList.querySelector(".script-list-operation")) {
            return;
          }
          let scriptInfo = parseScriptListInfo($scriptList);
          let $inlineStats = $scriptList.querySelector(
            ".inline-script-stats"
          );
          let code_url = scriptInfo.codeUrl;
          let $ratingScoreLeft = DOMUtils.createElement("dt", {
            className: "script-list-rating-score",
            innerHTML: `<span>${i18next.t("评分")}</span>`
          });
          let $ratingScoreRight = DOMUtils.createElement(
            "dd",
            {
              className: "script-list-rating-score",
              innerHTML: `<span>${scriptInfo.scriptRatingScore}</span>`
            },
            {
              "data-position": "right"
            }
          );
          let $goodRatingCount = $scriptList.querySelector(
            "dd.script-list-ratings .good-rating-count"
          );
          let $okRatingCount = $scriptList.querySelector(
            "dd.script-list-ratings .ok-rating-count"
          );
          let $badRatingCount = $scriptList.querySelector(
            "dd.script-list-ratings .bad-rating-count"
          );
          if ($goodRatingCount && $okRatingCount && $badRatingCount) {
            let goodRatingCount = parseInt($goodRatingCount.innerText);
            let okRatingCount = parseInt($okRatingCount.innerText);
            let badRatingCount = parseInt($badRatingCount.innerText);
            let totalRatingCount = goodRatingCount + okRatingCount + badRatingCount;
            if (totalRatingCount >= 10) {
              if (goodRatingCount / totalRatingCount >= 0.6) {
                $ratingScoreRight.classList.add("good-rating-count");
              } else {
                $ratingScoreRight.classList.add("bad-rating-count");
              }
            } else if (totalRatingCount == 0) {
              $ratingScoreRight.classList.add("good-rating-count");
            } else {
              if (goodRatingCount > okRatingCount + badRatingCount) {
                $ratingScoreRight.classList.add("good-rating-count");
              } else {
                $ratingScoreRight.classList.add("bad-rating-count");
              }
            }
          }
          let $versionLeft = DOMUtils.createElement("dt", {
            className: "script-list-version",
            innerHTML: (
              /*html*/
              `<span>${i18next.t("版本")}</span>`
            )
          });
          let $versionRight = DOMUtils.createElement(
            "dd",
            {
              className: "script-list-version",
              innerHTML: (
                /*html*/
                `<span>${scriptInfo.scriptVersion}</span>`
              )
            },
            {
              "data-position": "right"
            }
          );
          let $operationLeft = DOMUtils.createElement("dt", {
            className: "script-list-operation",
            innerHTML: `<span>${i18next.t("操作")}</span>`
          });
          let $operationRight = DOMUtils.createElement(
            "dd",
            {
              className: "script-list-operation",
              innerHTML: (
                /*html*/
                `
						<a
							target="_blank"
							class="install-link"
							data-install-format="js"
							data-script-name="${scriptInfo.scriptName}"
							data-script-namespace=""
							data-script-version="${scriptInfo.scriptVersion}"
							data-update-label="${i18next.t("更新到 {{version}} 版本", {
                version: scriptInfo.scriptVersion
              })}"
							data-downgrade-label="${i18next.t("降级到 {{version}} 版本", {
                version: scriptInfo.scriptVersion
              })}"
							data-reinstall-label="${i18next.t("重新安装 {{version}} 版本", {
                version: scriptInfo.scriptVersion
              })}"
							href="${code_url}"></a>
						<button class="script-collect-btn">${i18next.t("收藏")}</button>
						`
              )
            },
            {
              "data-position": "right",
              style: "gap:10px;display: flex;flex-wrap: wrap;align-items: center;"
            }
          );
          let $collect = $operationRight.querySelector(
            ".script-collect-btn"
          );
          let $installLink = $operationRight.querySelector(".install-link");
          $installLink["data-script-info"] = scriptInfo;
          $installLink.classList.add("lum-lightbox-loader");
          if (scriptInfo.scriptType === "library") {
            $installLink.remove();
          }
          DOMUtils.on($collect, "click", (event) => {
            utils.preventEvent(event);
            GreasyforkScriptsCollectEvent(scriptInfo.scriptId);
          });
          if (PopsPanel.getValue("gf-scripts-filter-enable")) {
            let $filter = DOMUtils.createElement("button", {
              className: "script-filter-btn",
              innerHTML: i18next.t("过滤")
            });
            let attr_filter_key = "data-filter-key";
            let attr_filter_value = "data-filter-value";
            DOMUtils.on($filter, "click", (event) => {
              utils.preventEvent(event);
              let $dialog = __pops.alert({
                title: {
                  text: i18next.t("选择需要过滤的选项"),
                  position: "center"
                },
                content: {
                  text: (
                    /*html*/
                    `
									<button ${attr_filter_key}="scriptId" ${attr_filter_value}="^${scriptInfo.scriptId}$">${i18next.t("脚本id:{{text}}", {
                    text: scriptInfo.scriptId
                  })}</button>
									<button ${attr_filter_key}="scriptName" ${attr_filter_value}="^${utils.parseStringToRegExpString(
                    scriptInfo.scriptName
                  )}$">${i18next.t("脚本名:{{text}}", {
                    text: scriptInfo.scriptName
                  })}</button>
									`
                  ),
                  html: true
                },
                mask: {
                  enable: true,
                  clickEvent: {
                    toClose: true
                  }
                },
                width: "350px",
                height: "300px",
                drag: true,
                dragLimit: true,
                style: (
                  /*css*/
                  `
								.pops-alert-content{
									display: flex;
									flex-direction: column;
    								gap: 20px;
								}
								.pops-alert-content button{
									text-wrap: wrap;
									padding: 8px;
									height: auto;
									text-align: left;
								}
								`
                )
              });
              let $content = $dialog.$shadowRoot.querySelector(
                ".pops-alert-content"
              );
              scriptInfo.scriptAuthors.forEach((scriptAuthorInfo) => {
                let $authorIdButton = DOMUtils.createElement("button", {
                  innerHTML: i18next.t("作者id:{{text}}", {
                    text: scriptAuthorInfo.authorId
                  })
                });
                $authorIdButton.setAttribute(attr_filter_key, "scriptAuthorId");
                $authorIdButton.setAttribute(
                  attr_filter_value,
                  "^" + scriptAuthorInfo.authorId + "$"
                );
                let $authorNameButton = DOMUtils.createElement("button", {
                  innerHTML: i18next.t("作者名:{{text}}", {
                    text: scriptAuthorInfo.authorName
                  })
                });
                $authorNameButton.setAttribute(
                  attr_filter_key,
                  "scriptAuthorName"
                );
                $authorNameButton.setAttribute(
                  attr_filter_value,
                  "^" + utils.parseStringToRegExpString(scriptAuthorInfo.authorName) + "$"
                );
                $content.appendChild($authorIdButton);
                $content.appendChild($authorNameButton);
              });
              DOMUtils.on(
                $dialog.$shadowRoot,
                "click",
                `button[${attr_filter_key}]`,
                (event2) => {
                  utils.preventEvent(event2);
                  let $click = event2.target;
                  let key = $click.getAttribute(
                    attr_filter_key
                  );
                  let value = $click.getAttribute(attr_filter_value);
                  GreasyforkScriptsFilter.addValue(key, value);
                  $dialog.close();
                  GreasyforkScriptsFilter.filter();
                  Qmsg.success(i18next.t("添加成功"));
                }
              );
            });
            $operationRight.appendChild($filter);
          }
          $inlineStats.appendChild($ratingScoreLeft);
          $inlineStats.appendChild($ratingScoreRight);
          $inlineStats.appendChild($versionLeft);
          $inlineStats.appendChild($versionRight);
          $inlineStats.appendChild($operationLeft);
          $inlineStats.appendChild($operationRight);
        });
        let $installLinkList = Array.from(
          document.querySelectorAll(
            ".install-link[data-install-format=js]"
          )
        );
        let scriptContainerStatus = GreasyforkCheckVersion.getScriptContainerStatus();
        let hasScriptContainer = Object.values(scriptContainerStatus).filter(
          Boolean
        );
        if (!scriptContainerStatus.Tampermonkey) {
          $installLinkList.forEach(async ($installLink) => {
            let result2 = await GreasyforkCheckVersion.checkForUpdatesJS(
              $installLink,
              true
            );
            $installLink.classList.remove("lum-lightbox-loader");
            if (!result2) {
              $installLink.textContent = i18next.t("安装此脚本");
            }
          });
        } else if (hasScriptContainer.length) {
          for (let index = 0; index < $installLinkList.length; index++) {
            let $installLink = $installLinkList[index];
            let scriptInfo = $installLink["data-script-info"];
            let getResp = await httpx.get(
              GreasyforkUrlUtils.getScriptInfoUrl(scriptInfo.scriptId),
              {
                fetch: true
              }
            );
            if (!getResp.status) {
              $installLink.textContent = i18next.t("安装此脚本");
              continue;
            }
            let data = utils.toJSON(
              getResp.data.responseText
            );
            $installLink.setAttribute("data-script-namespace", data.namespace);
            let result2 = await GreasyforkCheckVersion.checkForUpdatesJS(
              $installLink,
              true
            );
            $installLink.classList.remove("lum-lightbox-loader");
            if (!result2) {
              $installLink.textContent = i18next.t("安装此脚本");
            }
            await utils.sleep(150);
          }
        } else {
          log.error("未知的脚本容器");
        }
      });
      return result;
    }
  };
  const GreasyforkScriptsFilter = {
    /** 存储的键 */
    key: "gf-shield-rule",
    init() {
      log.info("脚本过滤");
      let lockFunction = new utils.LockFunction(() => {
        this.filter();
      }, 50);
      domUtils.ready(() => {
        utils.mutationObserver(document.body, {
          config: {
            subtree: true,
            childList: true
          },
          callback: () => {
            lockFunction.run();
          }
        });
        lockFunction.run();
      });
    },
    /**
     * 获取脚本列表元素
     */
    getElementList() {
      let scriptList = [];
      scriptList = scriptList.concat(
        Array.from(document.querySelectorAll("ol.script-list li"))
      );
      return scriptList;
    },
    /**
     * 对页面进行过滤
     */
    filter() {
      this.getElementList().forEach(($scriptList) => {
        let data = parseScriptListInfo($scriptList);
        let localValueSplit = this.getValue().split("\n");
        for (let index = 0; index < localValueSplit.length; index++) {
          let localRule = localValueSplit[index];
          let ruleSplit = localRule.split("##");
          let ruleName = ruleSplit[0];
          let ruleValue = ruleSplit[1];
          if (ruleName === "scriptRatingScore") {
            let userRatingScoreValue = parseFloat(ruleValue.slice(1));
            if (ruleValue.startsWith(">")) {
              if (data.scriptRatingScore > userRatingScoreValue) {
                log.info(["触发脚本过滤规则", [localRule, data]]);
                $scriptList.remove();
                break;
              }
            } else if (ruleValue.startsWith("<")) {
              if (data.scriptRatingScore < userRatingScoreValue) {
                log.info(["触发脚本过滤规则", [localRule, data]]);
                $scriptList.remove();
                break;
              }
            }
          } else if (ruleName in data || ruleName === "scriptDescription") {
            if (typeof ruleValue !== "string") {
              continue;
            }
            let ruleValueRegExp = new RegExp(ruleValue, "ig");
            let scriptInfoString = String(data[ruleName]);
            if (scriptInfoString.match(ruleValueRegExp)) {
              log.info(["触发脚本过滤规则", localRule, data]);
              $scriptList.remove();
              break;
            }
          }
        }
      });
    },
    setValue(value) {
      PopsPanel.setValue(this.key, value);
    },
    addValue(key, value) {
      let localValue = this.getValue();
      if (localValue.trim() !== "") {
        localValue += "\n";
      }
      if (utils.isNull(key)) {
        return;
      }
      key = key.toString().trim();
      let rule = key + "##" + value;
      localValue += rule;
      this.setValue(localValue);
    },
    getValue() {
      return PopsPanel.getValue(this.key, "");
    }
  };
  const SettingUIScripts = {
    id: "greasy-fork-panel-config-scripts",
    title: i18next.t("脚本"),
    forms: [
      {
        text: "",
        type: "forms",
        forms: [
          {
            text: i18next.t("代码"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("添加复制代码按钮"),
                    "addCopyCodeButton",
                    true,
                    void 0,
                    i18next.t("更优雅的复制")
                  ),
                  UISwitch(
                    i18next.t("快捷键"),
                    "fullScreenOptimization",
                    true,
                    void 0,
                    i18next.t("【F】键全屏、【Alt+Shift+F】键宽屏")
                  ),
                  UISwitch(
                    i18next.t("修复代码行号显示"),
                    "code-repairCodeLineNumber",
                    true,
                    void 0,
                    i18next.t("修复代码行数超过1k行号显示不全问题")
                  )
                ]
              }
            ]
          },
          {
            text: i18next.t("历史版本"),
            type: "deepMenu",
            forms: [
              {
                text: i18next.t("功能"),
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("添加额外的标签按钮"),
                    "scripts-versions-addExtraTagButton",
                    true,
                    void 0,
                    i18next.t("在版本下面添加【安装】、【查看代码】按钮")
                  )
                ]
              },
              {
                text: i18next.t("美化"),
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("美化历史版本页面"),
                    "beautifyHistoryVersionPage",
                    true,
                    void 0,
                    i18next.t("更直观的查看版本迭代")
                  )
                ]
              }
            ]
          }
        ]
      },
      {
        text: "",
        type: "forms",
        forms: [
          {
            text: i18next.t("功能"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("添加【寻找引用】按钮"),
                    "addFindReferenceButton",
                    true,
                    void 0,
                    i18next.t("在脚本栏添加按钮,一般用于搜索引用该库的相关脚本")
                  ),
                  UISwitch(
                    i18next.t("添加【收藏】按钮"),
                    "addCollectionButton",
                    true,
                    void 0,
                    i18next.t("在脚本栏添加按钮,一般用于快捷收藏该脚本/库")
                  ),
                  UISwitch(
                    i18next.t("添加【今日检查】信息块"),
                    "scriptHomepageAddedTodaySUpdate",
                    true,
                    void 0,
                    i18next.t("在脚本信息栏添加【今日检查】信息块")
                  )
                ]
              }
            ]
          },
          {
            text: i18next.t("美化"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("美化脚本列表"),
                    "beautifyCenterContent",
                    true,
                    void 0,
                    i18next.t("双列显示且添加脚本卡片操作项(安装、收藏)")
                  )
                ]
              }
            ]
          },
          {
            text: i18next.t("过滤"),
            type: "deepMenu",
            forms: [
              {
                text: `<a target="_blank" href="https://greasyfork.org/zh-CN/scripts/475722-greasyfork%E4%BC%98%E5%8C%96#:~:text=%E8%84%9A%E6%9C%AC%E8%BF%87%E6%BB%A4%E8%A7%84%E5%88%99">${i18next.t(
                "帮助文档"
              )}</a>`,
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("启用"),
                    "gf-scripts-filter-enable",
                    true,
                    void 0,
                    i18next.t("作用域:脚本、脚本搜索、用户主页")
                  ),
                  {
                    type: "own",
                    getLiElementCallBack(liElement) {
                      let textareaDiv = domUtils.createElement(
                        "div",
                        {
                          className: "pops-panel-textarea",
                          innerHTML: `
												<textarea placeholder="${i18next.t(
                          "请输入规则,每行一个"
                        )}" style="height:200px;"></textarea>`
                        },
                        {
                          style: "width: 100%;"
                        }
                      );
                      let $textarea = textareaDiv.querySelector(
                        "textarea"
                      );
                      $textarea.value = GreasyforkScriptsFilter.getValue();
                      domUtils.on(
                        $textarea,
                        ["input", "propertychange"],
                        void 0,
                        utils.debounce(function(event) {
                          GreasyforkScriptsFilter.setValue($textarea.value);
                        }, 200)
                      );
                      liElement.appendChild(textareaDiv);
                      return liElement;
                    }
                  }
                ]
              }
            ]
          }
        ]
      }
    ]
  };
  const GreasyforkDiscussionsFilter = {
    /** 存储的键 */
    key: "gf-discuessions-filter-rule",
    $data: {
      /** 脚本 */
      FILTER_SCRIPT_KEY: "greasyfork-discussions-filter-script",
      /** 发布用户 */
      FILTER_POST_USER_KEY: "greasyfork-discussions-filter-post-user",
      /** 回复用户 */
      FILTER_REPLY_USER_KEY: "greasyfork-discussions-filter-reply-user"
    },
    init() {
      log.info("论坛-过滤");
      _GM_addStyle(
        /*css*/
        `
        .discussion-list-container {
          --discusstion-repeat-color: #ffa700;
        }
        
        .discussion-list-container a.discussion-title[data-repeat-tip-show]::before {
          content: attr(data-repeat-tip-show);
          color: var(--discusstion-repeat-color);
          border-radius: 5px;
          border: 2px solid var(--discusstion-repeat-color);
          padding: 2px 5px;
          font-weight: 800;
          font-size: 14px;
        }
        `
      );
      let lockFunction = new utils.LockFunction(() => {
        this.filter();
      }, 50);
      utils.mutationObserver(document.body, {
        config: {
          subtree: true,
          childList: true
        },
        callback: () => {
          lockFunction.run();
        }
      });
      lockFunction.run();
    },
    /**
     * 获取反馈列表元素
     */
    getElementList() {
      let discussionsListContainer = [];
      discussionsListContainer = discussionsListContainer.concat(
        Array.from(
          document.querySelectorAll(".discussion-list-container")
        )
      );
      return discussionsListContainer;
    },
    /**
     * 论坛-过滤
     */
    filter() {
      this.transformOldRule();
      const SNIPPET_MAP = /* @__PURE__ */ new Map();
      this.getElementList().forEach(($listContainer, index) => {
        const discussionInfo = this.parseDiscuessionListContainerInfo($listContainer);
        let localValueSplit = this.getValue().split("\n");
        if (SNIPPET_MAP.has(discussionInfo.snippet) && PopsPanel.getValue("greasyfork-discussions-filter-duplicate-comments")) {
          let discussionTitleElement = SNIPPET_MAP.get(
            discussionInfo.snippet
          ).querySelector("a.discussion-title");
          discussionTitleElement.setAttribute("data-repeat-tip-show", "true");
          let oldCount = 0;
          if (discussionTitleElement.hasAttribute("data-repeat-count")) {
            oldCount = parseInt(
              discussionTitleElement.getAttribute("data-repeat-count")
            );
          }
          oldCount++;
          discussionTitleElement.setAttribute(
            "data-repeat-count",
            oldCount.toString()
          );
          discussionTitleElement.setAttribute(
            "data-repeat-tip-show",
            i18next.t("已过滤:{{oldCount}}", { oldCount })
          );
          log.success([
            `过滤重复内容:${discussionInfo.snippet}`,
            discussionInfo
          ]);
          $listContainer.remove();
          return;
        }
        SNIPPET_MAP.set(discussionInfo.snippet, $listContainer);
        for (let index2 = 0; index2 < localValueSplit.length; index2++) {
          let localRule = localValueSplit[index2];
          let ruleSplit = localRule.split("##");
          let ruleName = ruleSplit[0];
          let ruleValue = ruleSplit[1];
          if (ruleName in discussionInfo) {
            let ruleValueRegExp = new RegExp(ruleValue, "ig");
            if (discussionInfo[ruleName] != null) {
              let scriptInfoString = String(discussionInfo[ruleName]);
              if (scriptInfoString.match(ruleValueRegExp)) {
                log.info(["触发论坛过滤规则", localRule, discussionInfo]);
                $listContainer.remove();
                return;
              }
            }
          }
        }
      });
    },
    /**
     * 解析出元素上的属性
     */
    parseDiscuessionListContainerInfo($listContainer) {
      var _a2, _b, _c, _d;
      const info = {
        /** 脚本名 */
        scriptName: $listContainer.querySelector(
          ".discussion-meta-item-script-name"
        ).innerText,
        /** 脚本主页地址 */
        scriptUrl: (_a2 = $listContainer.querySelector(
          ".discussion-meta-item-script-name a"
        )) == null ? void 0 : _a2.href,
        /** 脚本id */
        scriptId: GreasyforkApi.getScriptId(
          (_b = $listContainer.querySelector(
            ".discussion-meta-item-script-name a"
          )) == null ? void 0 : _b.href
        ),
        /** 发布的用户名 */
        postUserName: $listContainer.querySelector("a.user-link").innerText,
        /** 发布的用户主页地址 */
        postUserHomeUrl: $listContainer.querySelector("a.user-link").href,
        /** 发布的用户id */
        postUserId: GreasyforkApi.getUserId(
          $listContainer.querySelector("a.user-link").href
        ),
        /** 发布的时间 */
        postTimeStamp: new Date(
          $listContainer.querySelector("relative-time").getAttribute("datetime")
        ),
        /** 发布的地址*/
        snippetUrl: $listContainer.querySelector("a.discussion-title").href,
        /** 发布的内容片段*/
        snippet: ((_c = $listContainer.querySelector("span.discussion-snippet")) == null ? void 0 : _c.innerText) || "",
        /** (如果有)回复的用户名*/
        replyUserName: void 0,
        /** (如果有)回复的用户主页地址*/
        replyUserHomeUrl: void 0,
        /** (如果有)回复的用户id*/
        replyUserId: void 0,
        /** (如果有)回复的时间 */
        replyTimeStamp: void 0
      };
      if ($listContainer.querySelector(
        ".discussion-meta-item .discussion-meta-item"
      )) {
        info.replyUserName = $listContainer.querySelector(
          ".discussion-meta-item .discussion-meta-item a.user-link"
        ).innerText;
        info.replyUserHomeUrl = $listContainer.querySelector(
          ".discussion-meta-item .discussion-meta-item a.user-link"
        ).href;
        info.replyUserId = GreasyforkApi.getUserId(info.replyUserHomeUrl);
        info.replyTimeStamp = new Date(
          (_d = $listContainer.querySelector(
            ".discussion-meta-item .discussion-meta-item relative-time"
          )) == null ? void 0 : _d.getAttribute("datetime")
        );
      }
      return info;
    },
    /** 转换旧规则 @deprecated */
    transformOldRule() {
      if (Date.now() > (/* @__PURE__ */ new Date("2024-8-19")).getTime()) {
        return;
      }
      const FILTER_SCRIPT_KEY = "greasyfork-discussions-filter-script";
      const FILTER_POST_USER_KEY = "greasyfork-discussions-filter-post-user";
      const FILTER_REPLY_USER_KEY = "greasyfork-discussions-filter-reply-user";
      const filterScript = PopsPanel.getValue(FILTER_SCRIPT_KEY, "");
      const filterPostUser = PopsPanel.getValue(FILTER_POST_USER_KEY, "");
      const filterReplyUser = PopsPanel.getValue(FILTER_REPLY_USER_KEY, "");
      const filterScriptList = filterScript.trim() === "" ? [] : filterScript.split("\n");
      const filterPostUserList = filterPostUser.trim() === "" ? [] : filterPostUser.split("\n");
      const filterReplyUserList = filterReplyUser.trim() === "" ? [] : filterReplyUser.split("\n");
      filterScriptList.forEach((ruleValue) => {
        this.addValue(
          "scriptId",
          utils.parseStringToRegExpString("^" + ruleValue + "$")
        );
      });
      filterPostUserList.forEach((ruleValue) => {
        this.addValue(
          "postUserId",
          utils.parseStringToRegExpString("^" + ruleValue + "$")
        );
      });
      filterReplyUserList.forEach((ruleValue) => {
        this.addValue(
          "replyUserId",
          utils.parseStringToRegExpString("^" + ruleValue + "$")
        );
      });
      PopsPanel.deleteValue(FILTER_SCRIPT_KEY);
      PopsPanel.deleteValue(FILTER_POST_USER_KEY);
      PopsPanel.deleteValue(FILTER_REPLY_USER_KEY);
    },
    setValue(value) {
      PopsPanel.setValue(this.key, value);
    },
    addValue(key, value) {
      let localValue = this.getValue();
      if (localValue.trim() !== "") {
        localValue += "\n";
      }
      if (utils.isNull(key)) {
        return;
      }
      key = key.toString().trim();
      let rule = key + "##" + value;
      localValue += rule;
      this.setValue(localValue);
    },
    getValue() {
      return PopsPanel.getValue(this.key, "");
    }
  };
  const SettingUIDiscuessions = {
    id: "greasy-fork-panel-config-discussions",
    title: i18next.t("论坛"),
    forms: [
      {
        text: "",
        type: "forms",
        forms: [
          {
            text: i18next.t("功能"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  {
                    type: "own",
                    attributes: {
                      "data-key": "discussions-readBgColor",
                      "data-default-value": "#e5e5e5"
                    },
                    getLiElementCallBack(liElement) {
                      let key = "discussions-readBgColor";
                      let $left = domUtils.createElement("div", {
                        className: "pops-panel-item-left-text",
                        innerHTML: `
											<p class="pops-panel-item-left-main-text">${i18next.t("自定义已读颜色")}</p>
											<p class="pops-panel-item-left-desc-text">${i18next.t("在讨论内生效")}</p>
											`
                      });
                      let $right = domUtils.createElement("div", {
                        className: "pops-panel-item-right",
                        innerHTML: `
											<input type="color" class="pops-color-choose" />
											`
                      });
                      let $color = $right.querySelector(
                        ".pops-color-choose"
                      );
                      $color.value = PopsPanel.getValue(key);
                      let $style = domUtils.createElement("style");
                      domUtils.append(document.head, $style);
                      domUtils.on(
                        $color,
                        ["input", "propertychange"],
                        (event) => {
                          log.info("选择颜色:" + $color.value);
                          $style.innerHTML = `
												.discussion-read{
													background: ${$color.value} !important;
												}
												`;
                          PopsPanel.setValue(key, $color.value);
                        }
                      );
                      liElement.appendChild($left);
                      liElement.appendChild($right);
                      return liElement;
                    }
                  },
                  UISwitch(
                    i18next.t("添加快捷操作按钮"),
                    "discussions-addShortcutOperationButton",
                    true,
                    void 0,
                    i18next.t(
                      "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效"
                    )
                  )
                ]
              }
            ]
          },
          {
            text: i18next.t("过滤"),
            type: "deepMenu",
            forms: [
              {
                text: `<a target="_blank" href="https://greasyfork.org/zh-CN/scripts/475722-greasyfork%E4%BC%98%E5%8C%96#:~:text=%E8%AE%BA%E5%9D%9B%E8%BF%87%E6%BB%A4%E8%A7%84%E5%88%99">${i18next.t(
                "帮助文档"
              )}</a>`,
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("启用"),
                    "greasyfork-discussions-filter-enable",
                    true,
                    void 0,
                    i18next.t("开启后下面的过滤功能才会生效")
                  ),
                  UISwitch(
                    i18next.t("过滤重复的评论"),
                    "greasyfork-discussions-filter-duplicate-comments",
                    false,
                    void 0,
                    i18next.t("过滤掉重复的评论数量(≥2)")
                  ),
                  {
                    type: "own",
                    getLiElementCallBack(liElement) {
                      let textareaDiv = domUtils.createElement(
                        "div",
                        {
                          className: "pops-panel-textarea",
                          innerHTML: `
										<textarea placeholder="${i18next.t(
                          "请输入规则,每行一个"
                        )}" style="height:200px;"></textarea>`
                        },
                        {
                          style: "width: 100%;"
                        }
                      );
                      let $textarea = textareaDiv.querySelector(
                        "textarea"
                      );
                      $textarea.value = GreasyforkDiscussionsFilter.getValue();
                      domUtils.on(
                        $textarea,
                        ["input", "propertychange"],
                        void 0,
                        utils.debounce(function(event) {
                          GreasyforkDiscussionsFilter.setValue($textarea.value);
                        }, 200)
                      );
                      liElement.appendChild(textareaDiv);
                      return liElement;
                    }
                  }
                ]
              }
            ]
          }
        ]
      }
    ]
  };
  const beautifyMarkdownCSS = 'code {\r\n	font-family: Menlo, Monaco, Consolas, "Courier New", monospace;\r\n	font-size: 0.85em;\r\n	color: #000;\r\n	background-color: #f0f0f0;\r\n	border-radius: 3px;\r\n	padding: 0.2em 0;\r\n}\r\ntable {\r\n	text-indent: initial;\r\n}\r\ntable {\r\n	margin: 10px 0 15px 0;\r\n	border-collapse: collapse;\r\n	border-spacing: 0;\r\n	display: block;\r\n	width: 100%;\r\n	overflow: auto;\r\n	word-break: normal;\r\n	word-break: keep-all;\r\n}\r\ncode,\r\npre {\r\n	color: #333;\r\n	background: 0 0;\r\n	font-family: Consolas, "Liberation Mono", Menlo, Courier, monospace;\r\n	text-align: left;\r\n	white-space: pre;\r\n	word-spacing: normal;\r\n	word-break: normal;\r\n	word-wrap: normal;\r\n	line-height: 1.4;\r\n	-moz-tab-size: 8;\r\n	-o-tab-size: 8;\r\n	tab-size: 8;\r\n	-webkit-hyphens: none;\r\n	-moz-hyphens: none;\r\n	-ms-hyphens: none;\r\n	hyphens: none;\r\n}\r\npre {\r\n	padding: 0.8em;\r\n	overflow: auto;\r\n	border-radius: 3px;\r\n	background: #f5f5f5;\r\n}\r\n:not(pre) > code {\r\n	padding: 0.1em;\r\n	border-radius: 0.3em;\r\n	white-space: normal;\r\n	background: #f5f5f5;\r\n}\r\nhtml body {\r\n	font-family: "Helvetica Neue", Helvetica, "Segoe UI", Arial, freesans,\r\n		sans-serif;\r\n	font-size: 16px;\r\n	line-height: 1.6;\r\n	color: #333;\r\n	background-color: #fff;\r\n	overflow: initial;\r\n	box-sizing: border-box;\r\n	word-wrap: break-word;\r\n}\r\nhtml body > :first-child {\r\n	margin-top: 0;\r\n}\r\nhtml body h1,\r\nhtml body h2,\r\nhtml body h3,\r\nhtml body h4,\r\nhtml body h5,\r\nhtml body h6 {\r\n	line-height: 1.2;\r\n	margin-top: 1em;\r\n	margin-bottom: 16px;\r\n	color: #000;\r\n}\r\nhtml body h1 {\r\n	font-size: 2.25em;\r\n	font-weight: 300;\r\n	padding-bottom: 0.3em;\r\n}\r\nhtml body h2 {\r\n	font-size: 1.75em;\r\n	font-weight: 400;\r\n	padding-bottom: 0.3em;\r\n}\r\nhtml body h3 {\r\n	font-size: 1.5em;\r\n	font-weight: 500;\r\n}\r\nhtml body h4 {\r\n	font-size: 1.25em;\r\n	font-weight: 600;\r\n}\r\nhtml body h5 {\r\n	font-size: 1.1em;\r\n	font-weight: 600;\r\n}\r\nhtml body h6 {\r\n	font-size: 1em;\r\n	font-weight: 600;\r\n}\r\nhtml body h1,\r\nhtml body h2,\r\nhtml body h3,\r\nhtml body h4,\r\nhtml body h5 {\r\n	font-weight: 600;\r\n}\r\nhtml body h5 {\r\n	font-size: 1em;\r\n}\r\nhtml body h6 {\r\n	color: #5c5c5c;\r\n}\r\nhtml body strong {\r\n	color: #000;\r\n}\r\nhtml body del {\r\n	color: #5c5c5c;\r\n}\r\nhtml body a:not([href]) {\r\n	color: inherit;\r\n}\r\nhtml body a {\r\n	text-decoration: underline;\r\n	text-underline-offset: 0.2rem;\r\n}\r\nhtml body a:hover {\r\n	color: #00a3f5;\r\n}\r\nhtml body img {\r\n	max-width: 100%;\r\n}\r\nhtml body > p {\r\n	margin-top: 0;\r\n	margin-bottom: 16px;\r\n	word-wrap: break-word;\r\n}\r\nhtml body > ol,\r\nhtml body > ul {\r\n	margin-bottom: 16px;\r\n}\r\nhtml body ol,\r\nhtml body ul {\r\n	padding-left: 2em;\r\n}\r\nhtml body ol.no-list,\r\nhtml body ul.no-list {\r\n	padding: 0;\r\n	list-style-type: none;\r\n}\r\nhtml body ol ol,\r\nhtml body ol ul,\r\nhtml body ul ol,\r\nhtml body ul ul {\r\n	margin-top: 0;\r\n	margin-bottom: 0;\r\n}\r\nhtml body li {\r\n	margin-bottom: 0;\r\n}\r\nhtml body li.task-list-item {\r\n	list-style: none;\r\n}\r\nhtml body li > p {\r\n	margin-top: 0;\r\n	margin-bottom: 0;\r\n}\r\nhtml body .task-list-item-checkbox {\r\n	margin: 0 0.2em 0.25em -1.8em;\r\n	vertical-align: middle;\r\n}\r\nhtml body .task-list-item-checkbox:hover {\r\n	cursor: pointer;\r\n}\r\nhtml body blockquote {\r\n	margin: 16px 0;\r\n	font-size: inherit;\r\n	padding: 0 15px;\r\n	color: #5c5c5c;\r\n	background-color: #f0f0f0;\r\n	border-left: 4px solid #d6d6d6 !important;\r\n}\r\nhtml body blockquote > :first-child {\r\n	margin-top: 0;\r\n}\r\nhtml body blockquote > :last-child {\r\n	margin-bottom: 0;\r\n}\r\nhtml body hr {\r\n	height: 4px;\r\n	margin: 32px 0;\r\n	background-color: #d6d6d6;\r\n	border: 0 none;\r\n}\r\nhtml body table {\r\n	margin: 10px 0 15px 0;\r\n	border-collapse: collapse;\r\n	border-spacing: 0;\r\n	display: block;\r\n	width: 100%;\r\n	overflow: auto;\r\n	word-break: normal;\r\n	word-break: keep-all;\r\n}\r\nhtml body table th {\r\n	font-weight: 700;\r\n	color: #000;\r\n}\r\nhtml body table td,\r\nhtml body table th {\r\n	border: 1px solid #d6d6d6;\r\n	padding: 6px 13px;\r\n}\r\nhtml body dl {\r\n	padding: 0;\r\n}\r\nhtml body dl dt {\r\n	padding: 0;\r\n	margin-top: 16px;\r\n	font-size: 1em;\r\n	font-style: italic;\r\n	font-weight: 700;\r\n}\r\nhtml body dl dd {\r\n	padding: 0 16px;\r\n	margin-bottom: 16px;\r\n}\r\nhtml body code {\r\n	font-family: Menlo, Monaco, Consolas, "Courier New", monospace;\r\n	font-size: 0.85em;\r\n	color: #000;\r\n	background-color: #f0f0f0;\r\n	border-radius: 3px;\r\n	padding: 0.2em 0;\r\n}\r\nhtml body code::after,\r\nhtml body code::before {\r\n	letter-spacing: -0.2em;\r\n	content: "\\00a0";\r\n}\r\nhtml body pre > code {\r\n	padding: 0;\r\n	margin: 0;\r\n	word-break: normal;\r\n	white-space: pre;\r\n	background: 0 0;\r\n	border: 0;\r\n}\r\nhtml body .highlight {\r\n	margin-bottom: 16px;\r\n}\r\nhtml body .highlight pre,\r\nhtml body pre {\r\n	padding: 1em;\r\n	overflow: auto;\r\n	line-height: 1.45;\r\n	border: #d6d6d6;\r\n	border-radius: 3px;\r\n}\r\nhtml body .highlight pre {\r\n	margin-bottom: 0;\r\n	word-break: normal;\r\n}\r\nhtml body pre code,\r\nhtml body pre tt {\r\n	display: inline;\r\n	max-width: initial;\r\n	padding: 0;\r\n	margin: 0;\r\n	overflow: initial;\r\n	line-height: inherit;\r\n	word-wrap: normal;\r\n	background-color: transparent;\r\n	border: 0;\r\n}\r\nhtml body pre code:after,\r\nhtml body pre code:before,\r\nhtml body pre tt:after,\r\nhtml body pre tt:before {\r\n	content: normal;\r\n}\r\nhtml body blockquote,\r\nhtml body dl,\r\nhtml body ol,\r\nhtml body p,\r\nhtml body pre,\r\nhtml body ul {\r\n	margin-top: 0;\r\n	margin-bottom: 16px;\r\n}\r\nhtml body kbd {\r\n	color: #000;\r\n	border: 1px solid #d6d6d6;\r\n	border-bottom: 2px solid #c7c7c7;\r\n	padding: 2px 4px;\r\n	background-color: #f0f0f0;\r\n	border-radius: 3px;\r\n}\r\n@media print {\r\n	html body {\r\n		background-color: #fff;\r\n	}\r\n	html body h1,\r\n	html body h2,\r\n	html body h3,\r\n	html body h4,\r\n	html body h5,\r\n	html body h6 {\r\n		color: #000;\r\n		page-break-after: avoid;\r\n	}\r\n	html body blockquote {\r\n		color: #5c5c5c;\r\n	}\r\n	html body pre {\r\n		page-break-inside: avoid;\r\n	}\r\n	html body table {\r\n		display: table;\r\n	}\r\n	html body img {\r\n		display: block;\r\n		max-width: 100%;\r\n		max-height: 100%;\r\n	}\r\n	html body code,\r\n	html body pre {\r\n		word-wrap: break-word;\r\n		white-space: pre;\r\n	}\r\n}\r\n/* 强制换行 */\r\ncode {\r\n	text-wrap: wrap !important;\r\n}\r\n\r\n.scrollbar-style::-webkit-scrollbar {\r\n	width: 8px;\r\n}\r\n.scrollbar-style::-webkit-scrollbar-track {\r\n	border-radius: 10px;\r\n	background-color: transparent;\r\n}\r\n.scrollbar-style::-webkit-scrollbar-thumb {\r\n	border-radius: 5px;\r\n	background-color: rgba(150, 150, 150, 0.66);\r\n	border: 4px solid rgba(150, 150, 150, 0.66);\r\n	background-clip: content-box;\r\n}\r\n';
  const beautifyButtonCSS = '/* 美化按钮 */\r\ninput[type="submit"],\r\nbutton {\r\n	display: inline-flex;\r\n	justify-content: center;\r\n	align-items: center;\r\n	line-height: 1;\r\n	height: 32px;\r\n	white-space: nowrap;\r\n	cursor: pointer;\r\n	/* color: #606266; */\r\n	text-align: center;\r\n	box-sizing: border-box;\r\n	outline: none;\r\n	transition: 0.1s;\r\n	font-weight: 500;\r\n	user-select: none;\r\n	vertical-align: middle;\r\n	appearance: none;\r\n	-webkit-appearance: none;\r\n	background-color: #ffffff;\r\n	border: 1px solid #dcdfe6;\r\n	border-color: #dcdfe6;\r\n	padding: 8px 15px;\r\n	font-size: 14px;\r\n	border-radius: 4px;\r\n}\r\n\r\ninput[type="submit"]:hover,\r\ninput[type="submit"]:focus,\r\nbutton:hover,\r\nbutton:focus {\r\n	color: #409eff;\r\n	border-color: #c6e2ff;\r\n	background-color: #ecf5ff;\r\n	outline: none;\r\n}\r\n\r\ninput[type="url"] {\r\n	position: relative;\r\n	font-size: 14px;\r\n	display: inline-flex;\r\n	line-height: 32px;\r\n	box-sizing: border-box;\r\n	vertical-align: middle;\r\n	appearance: none;\r\n	-webkit-appearance: none;\r\n	/* color: #606266; */\r\n	padding: 0;\r\n	outline: none;\r\n	border: none;\r\n	background: none;\r\n	flex-grow: 1;\r\n	align-items: center;\r\n	justify-content: center;\r\n	padding: 1px 11px;\r\n	background-color: #ffffff;\r\n	background-image: none;\r\n	border-radius: 4px;\r\n	cursor: text;\r\n	transition: box-shadow 0.2s cubic-bezier(0.645, 0.045, 0.355, 1);\r\n	transform: translateZ(0);\r\n	box-shadow: 0 0 0 1px #dcdfe6 inset;\r\n\r\n	width: 100%;\r\n	width: -moz-available;\r\n	width: -webkit-fill-available;\r\n	width: fill-available;\r\n}\r\n\r\ninput[type="url"]::placeholder {\r\n	color: #a8abb2;\r\n}\r\n\r\ninput[type="url"]:hover {\r\n	box-shadow: 0 0 0 1px #c0c4cc inset;\r\n}\r\n\r\ninput[type="url"]:focus {\r\n	box-shadow: 0 0 0 1px #409eff inset;\r\n}\r\n';
  const beautifyRadioCSS = 'label.radio-label {\r\n	font-weight: 500;\r\n	position: relative;\r\n	cursor: pointer;\r\n	display: inline-flex;\r\n	align-items: center;\r\n	white-space: normal;\r\n	outline: none;\r\n	font-size: 14px;\r\n	user-select: none;\r\n	margin-right: 32px;\r\n	height: 32px;\r\n	padding: 4px;\r\n	border-radius: 4px;\r\n	box-sizing: border-box;\r\n}\r\nlabel:has(input[type="radio"]:checked),\r\nlabel:has(input[type="radio"]:checked) a {\r\n	color: #409eff;\r\n}\r\nlabel.radio-label input[type="radio"] {\r\n	margin-right: 4px;\r\n	width: 14px;\r\n	height: 14px;\r\n}\r\nlabel.radio-label input[type="radio"]:checked {\r\n	-webkit-appearance: none;\r\n	-moz-appearance: none;\r\n	appearance: none;\r\n	border-radius: 50%;\r\n	width: 14px;\r\n	height: 14px;\r\n	outline: none;\r\n	border: 4px solid #409eff;\r\n	cursor: pointer;\r\n}\r\nlabel.radio-label input[type="radio"]:checked + span {\r\n	color: #409eff;\r\n}\r\n';
  const beautifyInputCSS = 'input[type="search"],\r\ninput[type="text"],\r\ninput[type="password"] {\r\n	justify-content: center;\r\n	align-items: center;\r\n	/* line-height: 1; */\r\n	/* height: 32px; */\r\n	white-space: nowrap;\r\n	cursor: text;\r\n	text-align: center;\r\n	box-sizing: border-box;\r\n	outline: 0;\r\n	transition: 0.1s;\r\n	/* font-weight: 500; */\r\n	user-select: none;\r\n	-webkit-user-select: none;\r\n	-moz-user-select: none;\r\n	-ms-user-select: none;\r\n	vertical-align: middle;\r\n	-webkit-appearance: none;\r\n	appearance: none;\r\n	background-color: transparent;\r\n	border: 0;\r\n	padding: 8px 8px;\r\n	/* font-size: 14px; */\r\n	text-align: start;\r\n	/* width: 100%; */\r\n	flex: 1;\r\n	border: 1px solid #dcdfe6;\r\n	border-radius: 4px;\r\n}\r\ninput[type="search"]:hover,\r\ninput[type="text"]:hover,\r\ninput[type="password"]:hover {\r\n	box-shadow: 0 0 0 1px #c0c4cc;\r\n}\r\ninput[type="search"]:focus,\r\ninput[type="text"]:focus,\r\ninput[type="password"]:focus {\r\n	outline: 0;\r\n	border: 1px solid #409eff;\r\n	border-radius: 4px;\r\n	box-shadow: none;\r\n}\r\n';
  const beautifyTextAreaCSS = "textarea {\r\n	display: block;\r\n	position: relative;\r\n	/*vertical-align: bottom;*/\r\n	position: relative;\r\n	resize: vertical;\r\n	padding: 5px 11px;\r\n	line-height: 1.5;\r\n	box-sizing: border-box;\r\n	width: 100%;\r\n	font-size: inherit;\r\n	font-family: inherit;\r\n	/* color: #606266; */\r\n	background-color: #ffffff;\r\n	background-image: none;\r\n	appearance: none;\r\n	-webkit-appearance: none;\r\n	box-shadow: 0 0 0 1px #dcdfe6 inset;\r\n	border-radius: 4px;\r\n	transition: box-shadow 0.2s cubic-bezier(0.645, 0.045, 0.355, 1);\r\n	border: none;\r\n}\r\ntextarea:focus {\r\n	outline: none;\r\n	box-shadow: 0 0 0 1px #409eff inset;\r\n}\r\n";
  const beautifyUploadImageCSS = '/* 隐藏 添加: */\r\nlabel[for="discussion_comments_attributes_0_attachments"],\r\nlabel[for="comment_attachments"] {\r\n	display: none;\r\n}\r\ninput[type="file"] {\r\n	width: 100%;\r\n	font-size: 20px;\r\n	background: #e2e2e2;\r\n	padding: 40px 0px;\r\n	border-radius: 10px;\r\n	text-align-last: center;\r\n}\r\n';
  const compatibleBeautifyCSS = "#main-header {\r\n	background-color: #670000 !important;\r\n	background-image: linear-gradient(#670000, #990000) !important;\r\n}\r\n#site-nav-vue {\r\n	flex-wrap: wrap;\r\n	justify-content: flex-end;\r\n}\r\n.open-sidebar {\r\n	border-width: 1px;\r\n	border-radius: 3px;\r\n	margin-right: 0;\r\n}\r\ninput.search-submit {\r\n	transform: translateY(-5%) !important;\r\n	margin-left: 10px;\r\n}\r\n#script-content code {\r\n	word-wrap: break-word;\r\n}\r\n.code-container ::selection {\r\n	background-color: #3d4556 !important;\r\n}\r\n";
  const beautifyTopNavigationBarCSS = "#language-selector {\r\n	display: none;\r\n}\r\n@media screen and (min-width: 600px) {\r\n	body {\r\n		--header-height: 50px;\r\n		--el-gap: 20px;\r\n	}\r\n\r\n	header#main-header {\r\n		height: var(--header-height);\r\n		position: fixed;\r\n		top: 0;\r\n		width: 100%;\r\n		z-index: 55555;\r\n		padding: unset;\r\n		display: flex;\r\n		justify-content: space-around;\r\n	}\r\n\r\n	header#main-header + div {\r\n		margin-top: calc(var(--header-height) + 35px);\r\n	}\r\n\r\n	header#main-header .width-constraint {\r\n		display: flex;\r\n		align-items: center;\r\n		gap: var(--el-gap);\r\n		padding: unset;\r\n		margin: unset;\r\n		max-width: unset;\r\n	}\r\n\r\n	header#main-header a {\r\n		text-decoration: none;\r\n		text-wrap: nowrap;\r\n	}\r\n\r\n	header#main-header .sign-out-link a {\r\n		text-decoration: underline;\r\n	}\r\n\r\n	header#main-header #site-name {\r\n		display: flex;\r\n		align-items: center;\r\n	}\r\n\r\n	header#main-header #site-name img {\r\n		width: calc(var(--header-height) - 5px);\r\n		height: calc(var(--header-height) - 5px);\r\n	}\r\n\r\n	/* 隐藏Greasyfork文字 */\r\n	header#main-header #site-name-text {\r\n		display: none;\r\n	}\r\n\r\n	header#main-header #site-nav {\r\n		display: flex;\r\n		flex-direction: row-reverse;\r\n		align-items: center;\r\n		flex: 1;\r\n		justify-content: space-between;\r\n		height: 100%;\r\n		gap: var(--el-gap);\r\n	}\r\n\r\n	header#main-header #site-nav nav li {\r\n		padding: 0 0.5em;\r\n		display: flex;\r\n		align-items: center;\r\n		height: var(--header-height);\r\n		min-width: 30px;\r\n		justify-content: center;\r\n	}\r\n\r\n	header#main-header #site-nav nav li:hover {\r\n		background: #5f0101;\r\n	}\r\n\r\n	header#main-header #nav-user-info {\r\n		max-width: 150px;\r\n	}\r\n\r\n	header#main-header #nav-user-info > span {\r\n		flex: 1;\r\n	}\r\n\r\n	header#main-header #nav-user-info,\r\n	header#main-header #nav-user-info + nav {\r\n		position: unset;\r\n		width: unset;\r\n		display: flex;\r\n		flex-wrap: nowrap;\r\n		align-items: center;\r\n	}\r\n}\r\n";
  const GreasyforkBeautify = {
    init() {
      PopsPanel.execMenuOnce("beautifyPage", () => {
        return this.beautifyPageElement();
      });
      PopsPanel.execMenuOnce("beautifyGreasyforkBeautify", () => {
        return this.beautifyGreasyforkBeautify();
      });
      PopsPanel.execMenuOnce("beautifyUploadImage", () => {
        return this.beautifyUploadImage();
      });
      PopsPanel.execMenuOnce("beautifyTopNavigationBar", () => {
        return this.beautifyTopNavigationBar();
      });
    },
    /**
     * 美化页面元素
     */
    beautifyPageElement() {
      log.info("美化页面元素");
      let result = [];
      result.push(_GM_addStyle(beautifyMarkdownCSS));
      result.push(_GM_addStyle(beautifyButtonCSS));
      result.push(_GM_addStyle(beautifyRadioCSS));
      result.push(_GM_addStyle(beautifyInputCSS));
      result.push(_GM_addStyle(beautifyTextAreaCSS));
      result.push(
        _GM_addStyle(
          /*css*/
          `
			p:has(input[type="submit"][name="update-and-sync"]){
			  margin-top: 10px;
			}
			`
        )
      );
      domUtils.ready(function() {
        let markupChoiceELement = document.querySelector(
          'a[target="markup_choice"][href*="daringfireball.net"]'
        );
        if (markupChoiceELement) {
          markupChoiceELement.parentElement.replaceChild(
            domUtils.createElement("span", {
              textContent: "Markdown"
            }),
            markupChoiceELement
          );
        }
        if (globalThis.location.pathname.endsWith("/admin") && !document.querySelector('input[type="submit"][name="update-only"]')) {
          result.push(
            _GM_addStyle(
              /*css*/
              `
					.indented{
						padding-left: unset;
					}
					`
            )
          );
        }
      });
      return result;
    },
    /**
     * 美化 Greasyfork Beautify脚本
     */
    beautifyGreasyforkBeautify() {
      log.info("美化 Greasyfork Beautify脚本");
      let result = [];
      result.push(_GM_addStyle(compatibleBeautifyCSS));
      if (utils.isPhone()) {
        result.push(
          _GM_addStyle(
            /*css*/
            `
				section#script-info,
				section.text-content,
				div.width-constraint table.text-content.log-table{
					margin-top: 80px;
				}
				
				div.width-constraint div.sidebarred{
					padding-top: 80px;
				}
				div.width-constraint div.sidebarred .sidebar{
					top: 80px;
				}`
          )
        );
      } else {
        result.push(
          _GM_addStyle(
            /*css*/
            `
				section#script-info{
					margin-top: 10px;
				}`
          )
        );
      }
      return result;
    },
    /**
     * 美化上传图片
     */
    beautifyUploadImage() {
      log.info("美化上传图片");
      let result = [];
      result.push(_GM_addStyle(beautifyUploadImageCSS));
      domUtils.ready(() => {
        function clearErrorTip(element) {
          while (element.nextElementSibling) {
            element.parentElement.removeChild(element.nextElementSibling);
          }
        }
        let $fileInputList = document.querySelectorAll('input[type="file"]');
        $fileInputList.forEach(($input) => {
          if ($input.getAttribute("name") === "code_upload") {
            return;
          }
          if ($input.hasAttribute("accept") && $input.getAttribute("accept").includes("javascript")) {
            return;
          }
          domUtils.on($input, ["propertychange", "input"], function(event) {
            clearErrorTip(event.target);
            let chooseImageFiles = event.currentTarget.files;
            if (!chooseImageFiles) {
              return;
            }
            if (chooseImageFiles.length === 0) {
              return;
            }
            log.info(["选择的图片", chooseImageFiles]);
            if (chooseImageFiles.length > 5) {
              domUtils.after(
                $input,
                domUtils.createElement("p", {
                  textContent: i18next.t(`❌ 最多同时长传5张图片`)
                })
              );
            }
            let notAllowImage = [];
            Array.from(chooseImageFiles).forEach((imageFile) => {
              if (imageFile.size > 204800 || !imageFile.type.match(/png|jpg|jpeg|gif|apng|webp/i)) {
                notAllowImage.push(imageFile);
              }
            });
            if (notAllowImage.length === 0) {
              return;
            }
            notAllowImage.forEach((imageFile) => {
              domUtils.after(
                $input,
                domUtils.createElement("p", {
                  textContent: i18next.t("❌ 图片:{{name}} 大小:{{size}}", {
                    name: imageFile.name,
                    size: imageFile.size
                  })
                })
              );
            });
          });
        });
        let $textAreaSelectorString = [
          /* 某条反馈内的回复框 */
          "textarea#comment_text",
          /* 反馈页面的回复框 */
          "textarea.comment-entry"
        ];
        $textAreaSelectorString.forEach((selector) => {
          domUtils.on(selector, "paste", (event) => {
            log.info(["触发粘贴事件", event]);
            setTimeout(() => {
              domUtils.trigger($fileInputList, "input");
            }, 100);
          });
        });
      });
      return result;
    },
    /**
     * 美化顶部导航栏
     */
    beautifyTopNavigationBar() {
      log.info("美化顶部导航栏");
      let result = [];
      result.push(_GM_addStyle(beautifyTopNavigationBarCSS));
      if (window.outerWidth > 550) {
        result.push(CommonUtils.addBlockCSS(".with-submenu"));
        domUtils.ready(() => {
          let $siteNav = document.querySelector("#site-nav");
          let $siteNavNav = $siteNav.querySelector("nav");
          document.querySelectorAll(".with-submenu nav li").forEach(($ele) => {
            $siteNavNav.appendChild($ele);
          });
        });
      }
      return result;
    }
  };
  const GreasyforkAccount = {
    init() {
      PopsPanel.execMenu("autoLogin", () => {
        this.autoLogin();
      });
    },
    /**
     * 自动登录
     */
    autoLogin() {
      utils.waitNode("span.sign-in-link a[rel=nofollow]").then(async () => {
        let user = PopsPanel.getValue("user");
        let pwd = PopsPanel.getValue("pwd");
        if (utils.isNull(user)) {
          Qmsg.error(i18next.t("请先在菜单中录入账号"));
          return;
        }
        if (utils.isNull(pwd)) {
          Qmsg.error(i18next.t("请先在菜单中录入密码"));
          return;
        }
        let csrfToken = document.querySelector("meta[name='csrf-token']");
        if (!csrfToken) {
          Qmsg.error(i18next.t("获取csrf-token失败"));
          return;
        }
        let loginTip = Qmsg.loading(i18next.t("正在登录中..."));
        let postResp = await httpx.post(
          "https://greasyfork.org/zh-CN/users/sign_in",
          {
            fetch: true,
            data: encodeURI(
              `authenticity_token=${csrfToken.getAttribute(
              "content"
            )}&user[email]=${user}&user[password]=${pwd}&user[remember_me]=1&commit=登录`
            ),
            headers: {
              "Content-Type": "application/x-www-form-urlencoded"
            }
          }
        );
        loginTip.destroy();
        if (!postResp.status) {
          log.error(postResp);
          Qmsg.error(i18next.t("登录失败,请在控制台查看原因"));
          return;
        }
        let respText = postResp.data.responseText;
        let parseLoginHTMLNode = domUtils.parseHTML(respText, true, true);
        if (parseLoginHTMLNode.querySelectorAll(
          ".sign-out-link a[rel=nofollow][data-method='delete']"
        ).length) {
          Qmsg.success(i18next.t("登录成功,1s后自动跳转"));
          setTimeout(() => {
            window.location.reload();
          }, 1e3);
        } else {
          log.error(postResp);
          log.error(`当前账号:${user}`);
          log.error(`当前密码:${pwd}`);
          Qmsg.error(
            i18next.t("登录失败,可能是账号/密码错误,请在控制台查看原因")
          );
        }
      });
    }
  };
  const GreasyforkForum = {
    init() {
      this.readBgColor();
      domUtils.ready(() => {
        PopsPanel.execMenuOnce("greasyfork-discussions-filter-enable", () => {
          this.filterEnable();
        });
        PopsPanel.execMenuOnce("discussions-addShortcutOperationButton", () => {
          this.addShortcutOperationButton();
        });
      });
    },
    /**
     * 启用Greasyfork论坛过滤器
     */
    filterEnable() {
      log.info("启用Greasyfork论坛过滤器");
      GreasyforkDiscussionsFilter.init();
    },
    /**
     * 设置已读背景颜色
     */
    readBgColor() {
      log.info("设置已读背景颜色");
      let color = PopsPanel.getValue("discussions-readBgColor");
      _GM_addStyle(
        /*css*/
        `
        .discussion-read{
            background: ${color} !important;
        }
        `
      );
    },
    /**
     * 添加快捷操作按钮
     */
    addShortcutOperationButton() {
      log.info("添加快捷操作按钮");
      GreasyforkDiscussionsFilter.getElementList().forEach(($listContainer) => {
        if ($listContainer.querySelector(
          ".discussion-filter-button"
        )) {
          return;
        }
        let $listItem = $listContainer.querySelector(
          ".discussion-list-item"
        );
        let $meta = $listItem.querySelector(".discussion-meta");
        let $ownMetaItem = domUtils.createElement("div", {
          className: "discussion-meta-item",
          innerHTML: `
					<button class="discussion-filter-button">${i18next.t("过滤")}</button>
					`
        });
        let $button = $ownMetaItem.querySelector(
          ".discussion-filter-button"
        );
        $meta.appendChild($ownMetaItem);
        domUtils.on($button, "click", (event) => {
          var _a2, _b, _c;
          utils.preventEvent(event);
          const discussionInfo = GreasyforkDiscussionsFilter.parseDiscuessionListContainerInfo(
            $listContainer
          );
          let attr_filter_key = "data-filter-key";
          let attr_filter_value = "data-filter-value";
          let $dialog = __pops.alert({
            title: {
              text: i18next.t("选择需要过滤的选项"),
              position: "center",
              html: false
            },
            content: {
              text: (
                /*html*/
                `
								<button ${attr_filter_key}="scriptId" ${attr_filter_value}="^${discussionInfo.scriptId}$">${i18next.t("脚本id:{{text}}", {
                text: discussionInfo.scriptId
              })}</button>
								<button ${attr_filter_key}="scriptName" ${attr_filter_value}="^${utils.parseStringToRegExpString(
                discussionInfo.scriptName
              )}$">${i18next.t("脚本名:{{text}}", {
                text: discussionInfo.scriptName
              })}</button>
								<button ${attr_filter_key}="postUserId" ${attr_filter_value}="^${utils.parseStringToRegExpString(
                discussionInfo.postUserId
              )}$">${i18next.t("发布的用户id:{{text}}", {
                text: discussionInfo.postUserId
              })}</button>
							`
              ),
              html: true
            },
            mask: {
              enable: true,
              clickEvent: {
                toClose: true
              }
            },
            drag: true,
            dragLimit: true,
            width: "350px",
            height: "300px",
            style: (
              /*css*/
              `
						.pops-alert-content{
							display: flex;
							flex-direction: column;
							gap: 20px;
						}
						.pops-alert-content button{
							text-wrap: wrap;
							padding: 8px;
							height: auto;
							text-align: left;
						}
						`
            )
          });
          let $content = $dialog.$shadowRoot.querySelector(
            ".pops-alert-content"
          );
          if (discussionInfo.scriptId == null) {
            (_a2 = $content.querySelector(`button[${attr_filter_key}="scriptId"]`)) == null ? void 0 : _a2.remove();
          }
          if (discussionInfo.scriptName == null) {
            (_b = $content.querySelector(`button[${attr_filter_key}="scriptName"]`)) == null ? void 0 : _b.remove();
          }
          if (discussionInfo.postUserId == null) {
            (_c = $content.querySelector(`button[${attr_filter_key}="postUserId"]`)) == null ? void 0 : _c.remove();
          }
          if (discussionInfo.replyUserId != null) {
            let $replyUserIdButton = domUtils.createElement("button", {
              innerHTML: i18next.t("作者id:{{text}}", {
                text: discussionInfo.replyUserId
              })
            });
            $replyUserIdButton.setAttribute(attr_filter_key, "scriptAuthorId");
            $replyUserIdButton.setAttribute(
              attr_filter_value,
              "^" + discussionInfo.replyUserId + "$"
            );
            $content.appendChild($replyUserIdButton);
          }
          domUtils.on(
            $dialog.$shadowRoot,
            "click",
            `button[${attr_filter_key}]`,
            (event2) => {
              utils.preventEvent(event2);
              let $click = event2.target;
              let key = $click.getAttribute(
                attr_filter_key
              );
              let value = $click.getAttribute(attr_filter_value);
              GreasyforkDiscussionsFilter.addValue(key, value);
              $dialog.close();
              GreasyforkDiscussionsFilter.filter();
              Qmsg.success(i18next.t("添加成功"));
            }
          );
        });
      });
    }
  };
  const GreasyforkUsers = {
    init() {
      PopsPanel.execMenuOnce("users-changeConsoleToTopNavigator", () => {
        this.changeConsoleToTopNavigator();
      });
      PopsPanel.execMenuOnce("gf-scripts-filter-enable", () => {
        GreasyforkScriptsFilter.init();
      });
      PopsPanel.execMenuOnce("beautifyCenterContent", () => {
        return GreasyforkScriptsList.beautifyCenterContent();
      });
    },
    /**
     * 迁移【控制台】到顶部导航栏
     */
    changeConsoleToTopNavigator() {
      log.info("迁移【控制台】到顶部导航栏");
      CommonUtils.addBlockCSS("#about-user");
      domUtils.ready(() => {
        let $aboutUser = document.querySelector("#about-user");
        let $siteNav = document.querySelector("#site-nav nav");
        if (!$aboutUser) {
          log.error("#about-user元素不存在");
          return;
        }
        if (!$siteNav) {
          log.error("#site-nav nav元素不存在");
          return;
        }
        $aboutUser = $aboutUser.cloneNode(true);
        let $consoleNav = domUtils.createElement("li", {
          className: "scripts-console",
          innerHTML: `<a href="javascript:;">${i18next.t("控制台")}</a>`
        });
        domUtils.on($consoleNav, "click", (event) => {
          utils.preventEvent(event);
          let $drawer = __pops.drawer({
            title: {
              enable: false
            },
            content: {
              text: "",
              html: true
            },
            size: "auto",
            direction: "top",
            zIndex: utils.getMaxZIndex(100),
            style: (
              /*css*/
              `
                    .text-content{
                        list-style-type: none;
                        box-shadow: rgb(221, 221, 221) 0px 0px 5px;
                        background-color: rgb(255, 255, 255);
                        box-sizing: border-box;
                        border-width: 1px;
                        border-style: solid;
                        border-color: rgb(187, 187, 187);
                        border-image: initial;
                        border-radius: 5px;
                        margin: 14px 0px;
                        padding: 10px 40px;
                    }
                    a.report-link{
                        position: absolute;
                        right: 0px;
                        font-size: smaller;
                        margin-right: 16px;
                        margin-top: 8px;
                    }
                    `
            )
          });
          let $drawerContent = $drawer.$shadowRoot.querySelector(
            ".pops-drawer-content"
          );
          $drawerContent.appendChild($aboutUser);
        });
        $siteNav.appendChild($consoleNav);
      });
    }
  };
  const Greasyfork = {
    init() {
      PopsPanel.execMenu("checkPage", () => {
        this.checkPage();
      });
      GreasyforkBeautify.init();
      if (GreasyforkRouter.isScript()) {
        GreasyforkScripts.init();
      }
      if (GreasyforkRouter.isScriptList() || GreasyforkRouter.isScriptLibraryList() || GreasyforkRouter.isScriptCodeSearch()) {
        GreasyforkScriptsList.init();
      }
      if (GreasyforkRouter.isDiscuessions()) {
        GreasyforkForum.init();
      }
      if (GreasyforkRouter.isUserHome()) {
        GreasyforkUsers.init();
      }
      PopsPanel.execMenuOnce("scripts-addOperationPanelBtnWithNavigator", () => {
        this.addOperationPanelBtnWithNavigator();
      });
      domUtils.ready(() => {
        GreasyforkMenu.initEnv();
        GreasyforkAccount.init();
        GreasyforkMenu.handleLocalGotoCallBack();
        PopsPanel.execMenuOnce("fixImageWidth", () => {
          Greasyfork.fixImageWidth();
        });
        Greasyfork.languageSelectorLocale();
        PopsPanel.execMenuOnce("optimizeImageBrowsing", () => {
          Greasyfork.optimizeImageBrowsing();
        });
        PopsPanel.execMenuOnce("overlayBedImageClickEvent", () => {
          Greasyfork.overlayBedImageClickEvent();
        });
        if (!GreasyforkRouter.isCodeStrict()) {
          PopsPanel.execMenuOnce("addMarkdownCopyButton", () => {
            Greasyfork.addMarkdownCopyButton();
          });
        }
      });
    },
    /**
     * 修复图片宽度显示问题
     */
    fixImageWidth() {
      if (window.innerWidth < window.innerHeight) {
        log.info("修复图片显示问题");
        _GM_addStyle(
          /*css*/
          `
            img.lum-img{
                width: 100% !important;
                height: 100% !important;
            }
          `
        );
      }
    },
    /**
     * 优化图片浏览
     */
    optimizeImageBrowsing() {
      log.info("优化图片浏览");
      {
        _GM_addStyle(_GM_getResourceText("ViewerCSS"));
      }
      _GM_addStyle(
        /*css*/
        `
        @media (max-width: 460px) {
          .lum-lightbox-image-wrapper {
              display:flex;
              overflow: auto;
              -webkit-overflow-scrolling: touch
          }
      
          .lum-lightbox-caption {
              width: 100%;
              position: absolute;
              bottom: 0
          }
      
          .lum-lightbox-position-helper {
              margin: auto
          }
      
          .lum-lightbox-inner img {
              max-width:100%;
              max-height:100%;
          }
        }
        `
      );
      function viewIMG(imgList = [], _index_ = 0) {
        let viewerULNodeHTML = "";
        imgList.forEach((item) => {
          viewerULNodeHTML += `<li><img data-src="${item}" loading="lazy"></li>`;
        });
        let viewerULNode = domUtils.createElement("ul", {
          innerHTML: viewerULNodeHTML
        });
        let viewer = new Viewer(viewerULNode, {
          inline: false,
          url: "data-src",
          zIndex: utils.getMaxZIndex() + 100,
          hidden: () => {
            viewer.destroy();
          }
        });
        _index_ = _index_ < 0 ? 0 : _index_;
        viewer.view(_index_);
        viewer.zoomTo(1);
        viewer.show();
      }
      function getImgElementSrc(element) {
        return element.getAttribute("data-src") || element.getAttribute("src") || element.getAttribute("alt");
      }
      domUtils.on(
        document,
        "click",
        "img",
        function(event) {
          var _a2;
          let imgElement = event.target;
          if (((_a2 = imgElement.parentElement) == null ? void 0 : _a2.localName) === "a" && imgElement.hasAttribute("data-screenshots")) {
            return;
          }
          if (imgElement.closest(".viewer-container")) {
            return;
          }
          if (imgElement.closest(".lum-lightbox-position-helper")) {
            return;
          }
          let userContentElement = imgElement.closest(".user-content");
          let imgList = [];
          let imgIndex = 0;
          let imgElementList = [];
          let currentImgSrc = getImgElementSrc(imgElement);
          if (currentImgSrc == null ? void 0 : currentImgSrc.startsWith("https://img.shields.io")) {
            return;
          }
          if (userContentElement) {
            userContentElement.querySelectorAll("img").forEach((childImgElement) => {
              imgElementList.push(childImgElement);
              let imgSrc = getImgElementSrc(childImgElement);
              let $parent = childImgElement.parentElement;
              if (($parent == null ? void 0 : $parent.localName) === "a") {
                imgSrc = $parent.getAttribute("data-href") || $parent.href;
              }
              imgList.push(imgSrc);
            });
            imgIndex = imgElementList.indexOf(imgElement);
            if (imgIndex === -1) {
              imgIndex = 0;
            }
          } else {
            imgList.push(currentImgSrc);
            imgIndex = 0;
          }
          log.success(["点击浏览图片👉", imgList, imgIndex]);
          viewIMG(imgList, imgIndex);
        }
      );
      document.querySelectorAll(".user-screenshots").forEach((element) => {
        let linkElement = element.querySelector("a");
        if (!linkElement) {
          return;
        }
        let imgSrc = linkElement.getAttribute("data-href") || linkElement.getAttribute("href");
        let imgElement = element.querySelector("img");
        if (!imgElement) {
          return;
        }
        imgElement.setAttribute("data-screenshots", "true");
        imgElement.setAttribute("data-src", imgSrc);
        linkElement.setAttribute("href", "javascript:;");
        domUtils.after(linkElement, imgElement);
        linkElement.remove();
      });
    },
    /**
     * 覆盖图床图片的parentElement的a标签
     */
    overlayBedImageClickEvent() {
      log.info("覆盖图床图片的parentElement的a标签");
      document.querySelectorAll(".user-content a>img").forEach((imgElement) => {
        let linkElement = imgElement.parentElement;
        let url = linkElement.getAttribute("href");
        linkElement.setAttribute("data-href", url);
        linkElement.removeAttribute("href");
        domUtils.on(linkElement, "click", () => {
          Qmsg.warning(
            /*html*/
            `<div style="overflow-wrap: anywhere;">${i18next.t(
            "拦截跳转:"
          )}<a href="${url}" target="_blank">${url}</a></div>`,
            {
              html: true,
              zIndex: utils.getMaxZIndex() + 105
            }
          );
        });
      });
    },
    /**
     * 在Markdown右上角添加复制按钮
     */
    addMarkdownCopyButton() {
      log.info("在Markdown右上角添加复制按钮");
      _GM_addStyle(
        /*css*/
        `
        pre{
          position: relative;
          margin-bottom: 0px !important;
          width: 100%;
        }
        `
      );
      _GM_addStyle(
        /*css*/
        `
        .snippet-clipboard-content{
          display: flex;
          justify-content: space-between;
          background: rgb(246, 248, 250);
          margin-bottom: 16px;
        }
        .zeroclipboard-container {
          /* right: 0;
          top: 0;
          position: absolute; */
          box-sizing: border-box;
          display: flex;
          font-size: 16px;
          line-height: 24px;
          text-size-adjust: 100%;
          overflow-wrap: break-word;
          width: fit-content;
          height: fit-content;
        }
        .zeroclipboard-container svg{
            vertical-align: text-bottom;
            display: inline-block;
            overflow: visible;
            fill: currentColor;
            margin: 8px;
        }
        .zeroclipboard-container svg[aria-hidden="true"]{
          display: none;
        }
        clipboard-copy.js-clipboard-copy {
          position: relative;
          padding: 0px;
          color: rgb(36, 41, 47);
          background-color: rgb(246, 248, 250);
          transition: 80ms cubic-bezier(0.33, 1, 0.68, 1);
          transition-property: color,background-color,box-shadow,border-color;
          display: inline-block;
          font-size: 14px;
          line-height: 20px;
          white-space: nowrap;
          vertical-align: middle;
          cursor: pointer;
          -webkit-user-select: none;
          user-select: none;
          border: 1px solid rgba(31, 35, 40, 0.15);
          -webkit-appearance: none;
          appearance: none;
          box-shadow: rgba(31, 35, 40, 0.04) 0px 1px 0px 0px, rgba(255, 255, 255, 0.25) 0px 1px 0px 0px inset;
          margin: 8px;
          overflow-wrap: break-word;
          text-wrap: nowrap;
          border-radius: 6px;
        }
        clipboard-copy.js-clipboard-copy[success]{
          border-color: rgb(31, 136, 61);
          box-shadow: 0 0 0 0.2em rgba(52,208,88,.4);
        }
        clipboard-copy.js-clipboard-copy:hover{
          background-color: rgb(243, 244, 246);
          border-color: rgba(31, 35, 40, 0.15);
          transition-duration: .1s;
        }
        clipboard-copy.js-clipboard-copy:active{
          background-color: rgb(235, 236, 240);
          border-color: rgba(31, 35, 40, 0.15);
          transition: none;
        }
        `
      );
      _GM_addStyle(
        /*css*/
        `
        .pops-tip.github-tooltip {
          border-radius: 6px;
          padding: 6px 8px;
        }
        
        .pops-tip.github-tooltip, .pops-tip.github-tooltip .pops-tip-arrow::after {
          background: rgb(36, 41, 47);
          color: #fff;
        }
        
        .pops-tip.github-tooltip .pops-tip-arrow::after {
          width: 8px;
          height: 8px;
        }
        `
      );
      function getCopyElement() {
        let copyElement = domUtils.createElement("div", {
          className: "zeroclipboard-container",
          innerHTML: (
            /*html*/
            `
				<clipboard-copy class="js-clipboard-copy">
				<svg height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon-copy">
					<path d="M0 6.75C0 5.784.784 5 1.75 5h1.5a.75.75 0 0 1 0 1.5h-1.5a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 0 0 .25-.25v-1.5a.75.75 0 0 1 1.5 0v1.5A1.75 1.75 0 0 1 9.25 16h-7.5A1.75 1.75 0 0 1 0 14.25Z"></path><path d="M5 1.75C5 .784 5.784 0 6.75 0h7.5C15.216 0 16 .784 16 1.75v7.5A1.75 1.75 0 0 1 14.25 11h-7.5A1.75 1.75 0 0 1 5 9.25Zm1.75-.25a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 0 0 .25-.25v-7.5a.25.25 0 0 0-.25-.25Z"></path>
				</svg>
				<svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon-check-copy">
					<path d="M13.78 4.22a.75.75 0 0 1 0 1.06l-7.25 7.25a.75.75 0 0 1-1.06 0L2.22 9.28a.751.751 0 0 1 .018-1.042.751.751 0 0 1 1.042-.018L6 10.94l6.72-6.72a.75.75 0 0 1 1.06 0Z"></path>
				</svg>
				</clipboard-copy>
            `
          )
        });
        let clipboardCopyElement = copyElement.querySelector(
          ".js-clipboard-copy"
        );
        let octiconCopyElement = copyElement.querySelector(
          ".octicon-copy"
        );
        let octiconCheckCopyElement = copyElement.querySelector(
          ".octicon-check-copy"
        );
        domUtils.on(copyElement, "click", function() {
          let codeElement = copyElement.parentElement.querySelector("code");
          if (!codeElement && copyElement.parentElement.className.includes("prettyprinted")) {
            codeElement = copyElement.parentElement;
          }
          if (!codeElement) {
            Qmsg.error(i18next.t("未找到{{selector}}元素", { selector: "code" }));
            return;
          }
          utils.setClip(codeElement.innerText || codeElement.textContent);
          clipboardCopyElement.setAttribute("success", "true");
          octiconCopyElement.setAttribute("aria-hidden", "true");
          octiconCheckCopyElement.removeAttribute("aria-hidden");
          let tooltip = __pops.tooltip({
            target: clipboardCopyElement,
            content: i18next.t("✅ 复制成功!"),
            position: "left",
            className: "github-tooltip",
            alwaysShow: true
          });
          setTimeout(() => {
            clipboardCopyElement.removeAttribute("success");
            octiconCheckCopyElement.setAttribute("aria-hidden", "true");
            octiconCopyElement.removeAttribute("aria-hidden");
            tooltip.close();
          }, 2e3);
        });
        return copyElement;
      }
      document.querySelectorAll("pre").forEach((preElement) => {
        let zeroclipboardElement = preElement.querySelector(
          "div.zeroclipboard-container"
        );
        if (zeroclipboardElement) {
          return;
        }
        let copyElement = getCopyElement();
        let snippetClipboardContentElement = domUtils.createElement("div", {
          className: "snippet-clipboard-content"
        });
        domUtils.before(preElement, snippetClipboardContentElement);
        snippetClipboardContentElement.appendChild(preElement);
        snippetClipboardContentElement.appendChild(copyElement);
      });
    },
    /**
     * 固定当前语言
     */
    languageSelectorLocale() {
      let localeLanguage = PopsPanel.getValue("language-selector-locale");
      let currentLocaleLanguage = window.location.pathname.split("/").filter((item) => Boolean(item))[0];
      log.success("选择语言:" + localeLanguage);
      log.success("当前语言:" + currentLocaleLanguage);
      if (utils.isNull(localeLanguage)) {
        return;
      }
      if (localeLanguage === currentLocaleLanguage) {
        return;
      } else {
        let timer = null;
        let url = GreasyforkApi.getSwitchLanguageUrl(localeLanguage);
        GreasyforkApi.switchLanguage(url);
        log.success("新Url:" + url);
        Qmsg.loading(
          i18next.t(
            "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}",
            {
              currentLocaleLanguage,
              localeLanguage
            }
          ),
          {
            timeout: 3e3,
            showClose: true,
            onClose() {
              clearTimeout(timer);
            }
          }
        );
        Qmsg.info(i18next.t("导航至:") + url, {
          timeout: 3e3
        });
        timer = setTimeout(() => {
          window.location.href = url;
        }, 3e3);
      }
    },
    /**
     * 面板-脚本列表|库
     * @param type
     * @param event
     * @param rightHeaderElement
     * @param rightContainerElement
     * @returns
     */
    async UIScriptList(type, event, rightHeaderElement, rightContainerElement) {
      var _a2, _b, _c;
      if (!GreasyforkMenu.isLogin) {
        Qmsg.error(i18next.t("请先登录账号!"));
        return;
      }
      let userLinkElement = GreasyforkMenu.getUserLinkElement();
      let userLink = userLinkElement.href;
      let userId = (_c = (_b = (_a2 = userLink == null ? void 0 : userLink.split("/")) == null ? void 0 : _a2.pop()) == null ? void 0 : _b.match(/([0-9]+)/)) == null ? void 0 : _c[0];
      let loading = __pops.loading({
        mask: {
          enable: true
        },
        parent: rightContainerElement,
        content: {
          text: i18next.t("获取信息中,请稍后...")
        },
        addIndexCSS: false
      });
      let userInfo = await GreasyforkApi.getUserInfo(userId);
      loading.close();
      if (!userInfo) {
        return;
      }
      log.info(userInfo);
      let scriptList = type === "script-list" ? userInfo["scriptList"] : userInfo["scriptLibraryList"];
      Qmsg.success(
        i18next.t("获取成功,共 {{count}} 个", {
          count: scriptList.length
        })
      );
      for (const scriptInfo of scriptList) {
        let liElement = domUtils.createElement("li", {
          className: "w-script-list-item",
          innerHTML: (
            /*html*/
            `
				<div class="w-script-info">
				<div class="w-script-name">
					<a href="${scriptInfo["url"]}" target="_blank">${scriptInfo["name"]}</a>
				</div>
				<div class="w-script-fan-score">
					<p>${i18next.t("评分:")}${scriptInfo["fan_score"]}</p>
				</div>
				<div class="w-script-locale">
					<p>${i18next.t("语言:")}${scriptInfo["locale"]}</p>
				</div>
				<div class="w-script-version">
					<p>${i18next.t("版本:")}${scriptInfo["version"]}</p>
				</div>
				<div class="w-script-update-time">
					<p>${i18next.t("更新:")}${utils.getDaysDifference(
            new Date(scriptInfo["code_updated_at"]).getTime(),
            void 0,
            "auto"
          )}前</p>
				</div>
				</div>
            `
          )
        });
        let scriptInfoElement = liElement.querySelector(
          ".w-script-info"
        );
        let buttonElement = domUtils.createElement("div", {
          className: "pops-panel-button",
          innerHTML: (
            /*html*/
            `
				<button type="primary" data-icon="" data-righticon="false">
				<span>${i18next.t("同步代码")}</span>
				</button>
				`
          )
        });
        if (scriptInfo["deleted"]) {
          liElement.classList.add("w-script-deleted");
          buttonElement.querySelector("button").setAttribute("disabled", "true");
        }
        domUtils.on(buttonElement, "click", void 0, async function() {
          log.success(["同步", scriptInfo]);
          let btn = buttonElement.querySelector("button");
          let span = buttonElement.querySelector(
            "button span"
          );
          let iconElement = domUtils.createElement(
            "i",
            {
              className: "pops-bottom-icon",
              innerHTML: __pops.config.iconSVG.loading
            },
            {
              "is-loading": true
            }
          );
          btn.setAttribute("disabled", "true");
          btn.setAttribute("data-icon", "true");
          span.innerText = i18next.t("同步中...");
          domUtils.before(span, iconElement);
          let scriptId = scriptInfo == null ? void 0 : scriptInfo["id"];
          let codeSyncFormData = await GreasyforkApi.getSourceCodeSyncFormData(
            scriptId.toString()
          );
          if (codeSyncFormData) {
            const SCRIPT_SYNC_TYPE_ID_FORMDATA_KEY = "script[script_sync_type_id]";
            if (codeSyncFormData.has(SCRIPT_SYNC_TYPE_ID_FORMDATA_KEY)) {
              let syncTypeId = codeSyncFormData.get(
                SCRIPT_SYNC_TYPE_ID_FORMDATA_KEY
              );
              let syncMode = "";
              if (syncTypeId.toString() === "1") {
                syncMode = i18next.t("手动");
              } else if (syncTypeId.toString() === "2") {
                syncMode = i18next.t("自动");
              } else if (syncTypeId.toString() === "3") {
                syncMode = "webhook";
              }
              let oldSyncTypeElement = liElement.querySelector(
                ".w-script-sync-type"
              );
              if (oldSyncTypeElement) {
                oldSyncTypeElement.querySelector("p").innerText = i18next.t(
                  "同步方式:{{syncMode}}",
                  { syncMode }
                );
              } else {
                domUtils.append(
                  scriptInfoElement,
                  /*html*/
                  `
								<div class="w-script-sync-type">
									<p>${i18next.t("同步方式:{{syncMode}}", {
                  syncMode
                })}
									</p>
								</div>`
                );
              }
              let syncUpdateResponse = await GreasyforkApi.sourceCodeSync(
                scriptInfo["id"].toString(),
                codeSyncFormData
              );
              if (syncUpdateResponse) {
                Qmsg.success(i18next.t("同步成功"));
              } else {
                Qmsg.error(i18next.t("同步失败"));
              }
            } else {
              Qmsg.error(i18next.t("该脚本未设置同步信息"));
            }
          }
          btn.removeAttribute("disabled");
          btn.removeAttribute("data-icon");
          span.innerText = i18next.t("同步代码");
          iconElement.remove();
        });
        liElement.appendChild(buttonElement);
        rightContainerElement.appendChild(liElement);
      }
    },
    /**
     * 检测gf页面是否正确加载,有时候会出现
     * We're down for maintenance. Check back again soon.
     */
    checkPage() {
      log.info("检测gf页面是否正确加载,有时候会出现");
      domUtils.ready(() => {
        if (document.body.firstElementChild && document.body.firstElementChild.localName === "p" && document.body.firstElementChild.innerText.includes(
          "We're down for maintenance. Check back again soon."
        )) {
          let latestRefreshPageTime = parseInt(
            _GM_getValue(
              "greasyfork-check-page-time",
              0
            )
          );
          let checkPageTimeout = PopsPanel.getValue(
            "greasyfork-check-page-timeout",
            5
          );
          let checkPageTimeoutStamp = checkPageTimeout * 1e3;
          if (latestRefreshPageTime && Date.now() - latestRefreshPageTime < checkPageTimeoutStamp) {
            Qmsg.warning(
              i18next.t("上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载", {
                time: utils.formatTime(
                  latestRefreshPageTime,
                  "yyyy-MM-dd HH:mm:ss"
                ),
                timeout: checkPageTimeout
              })
            );
            return;
          }
          _GM_setValue("greasyfork-check-page-time", Date.now());
          window.location.reload();
        }
      });
    },
    /**
     * 在顶部导航栏添加【操作面板】按钮
     */
    addOperationPanelBtnWithNavigator() {
      log.info("添加【操作面板】按钮");
      CommonUtils.addBlockCSS(
        ".sidebarred .sidebar",
        ".sidebarred-main-content .open-sidebar"
      );
      _GM_addStyle(
        /*css*/
        `
		.sidebarred .sidebarred-main-content{
			max-width: 100%;
		}	
		`
      );
      domUtils.ready(() => {
        let $nav = document.querySelector("#site-nav nav");
        let $subNav = document.querySelector(
          "#site-nav .with-submenu nav"
        );
        let $scriptsOptionGroups = document.querySelector("#script-list-option-groups") || document.querySelector(".list-option-groups");
        if (!$scriptsOptionGroups) {
          log.warn("不存在右侧面板元素#script-list-option-groups");
          return;
        }
        $scriptsOptionGroups = $scriptsOptionGroups.cloneNode(
          true
        );
        $scriptsOptionGroups.classList.add("option-panel-groups");
        if (!$nav) {
          log.error("元素#site-nav nav不存在");
          return;
        }
        let $filterBtn = domUtils.createElement("li", {
          className: "filter-scripts",
          innerHTML: `
                <a href="javascript:;">${i18next.t("操作面板")}</a>
                `
        });
        domUtils.on($filterBtn, "click", (event) => {
          utils.preventEvent(event);
          let $drawer = __pops.drawer({
            title: {
              enable: false
            },
            content: {
              text: "",
              html: true
            },
            direction: "top",
            size: "80%",
            zIndex: utils.getMaxZIndex(100),
            style: (
              /*css*/
              `
                    .pops-drawer-content div:first-child{
                        margin: 20px 0 0 0;
                    }
                    .option-panel-groups > div{
                    
                    }
                    .option-panel-groups ul{
                        margin: .5em 0 0;
                        list-style-type: none;
                        padding: 1em 0;
                        box-shadow: 0 0 5px #ddd;
                        border: 1px solid #BBBBBB;
                        border-radius: 5px;
                        background-color: #fff;
                    }
                    .option-panel-groups ul li{
                    
                    }
                    li.list-current{
                        border-left: 7px solid #800;
                        box-shadow: inset 0 1px #0000001a, inset 0 -1px #0000001a;
                        margin: 0 0 0 -4px;
                        padding: .4em 1em .4em calc(1em - 3px);
                        background: linear-gradient(#fff, #eee);
                    }
                    .list-option-group a {
                        padding: .35em 1em;
                        display: block;
                    }
                    .list-option-group {
                        margin-bottom: 1em;
                    }
                    form.sidebar-search{
                        display: flex;
                        align-items: center;
                        gap: 10px;
                    }
                    form.sidebar-search input[type="search"]{
                        display: inline-flex;
                        justify-content: center;
                        align-items: center;
                        line-height: 1;
                        height: 32px;
                        white-space: nowrap;
                        cursor: text;
                        text-align: center;
                        box-sizing: border-box;
                        outline: 0;
                        transition: 0.1s;
                        font-weight: 500;
                        user-select: none;
                        -webkit-user-select: none;
                        -moz-user-select: none;
                        -ms-user-select: none;
                        vertical-align: middle;
                        -webkit-appearance: none;
                        appearance: none;
                        background-color: transparent;
                        border: 0;
                        padding: 8px 8px;
                        font-size: 14px;
                        text-align: start;
                        /* width: 100%; */
                        // flex: 1;
                        display: flex;
                        align-items: center;
                        border: 1px solid #dcdfe6;
                        border-radius: 4px;
                        background-color: #ffffff;
                    }
                    form.sidebar-search input[type="submit"]{
                        width: 32px;
                        height: 32px;
                    }
                    `
            )
          });
          let $drawerContent = $drawer.$shadowRoot.querySelector(
            ".pops-drawer-content"
          );
          $drawerContent.appendChild($scriptsOptionGroups);
        });
        if ($subNav && $subNav.children.length) {
          $subNav.appendChild($filterBtn);
        } else {
          $nav.appendChild($filterBtn);
        }
      });
    }
  };
  const SettingUIScriptList = {
    id: "greasy-fork-panel-config-script-list",
    title: i18next.t("脚本列表"),
    callback(event, rightHeaderElement, rightContainerElement) {
      Greasyfork.UIScriptList(
        "script-list",
        event,
        rightHeaderElement,
        rightContainerElement
      );
    },
    forms: []
  };
  const SettingUIScriptLib = {
    id: "greasy-fork-panel-config-library",
    title: i18next.t("库"),
    callback(event, rightHeaderElement, rightContainerElement) {
      Greasyfork.UIScriptList(
        "script-library",
        event,
        rightHeaderElement,
        rightContainerElement
      );
    },
    forms: []
  };
  const UIScriptListCSS = '.w-script-list-item {\r\n	padding: 10px 0;\r\n	border-bottom: 1px solid #e5e5e5;\r\n	font-size: 16px;\r\n	text-align: left;\r\n}\r\n.w-script-version,\r\n.w-script-fan-score,\r\n.w-script-create-time,\r\n.w-script-update-time,\r\n.w-script-locale,\r\n.w-script-sync-type {\r\n	font-size: 14px;\r\n	color: #7c7c7c;\r\n}\r\n.w-script-fan-score {\r\n	margin-left: unset !important;\r\n	text-align: unset !important;\r\n	max-width: unset !important;\r\n}\r\n.w-script-deleted {\r\n	text-decoration: line-through;\r\n	font-style: italic;\r\n	color: red;\r\n}\r\n.w-script-deleted .w-script-name::before {\r\n	content: "【删除】";\r\n}\r\n\r\nli[data-key="user"] .pops-panel-input,\r\nli[data-key="pwd"] .pops-panel-input {\r\n	max-width: 200px;\r\n}\r\n';
  const SettingUIUsers = {
    id: "greasy-fork-panel-config-account",
    title: i18next.t("用户"),
    forms: [
      {
        text: "",
        type: "forms",
        forms: [
          {
            text: i18next.t("功能"),
            type: "deepMenu",
            forms: [
              {
                text: "",
                type: "forms",
                forms: [
                  UISwitch(
                    i18next.t("迁移【控制台】到顶部导航栏"),
                    "users-changeConsoleToTopNavigator",
                    true,
                    void 0,
                    i18next.t("将【控制台】按钮移动到顶部导航栏,节省空间")
                  )
                ]
              }
            ]
          }
        ]
      }
    ]
  };
  const __PopsPanel__ = {
    data: null,
    oneSuccessExecMenu: null,
    onceExec: null,
    listenData: null
  };
  const PopsPanel = {
    /** 数据 */
    $data: {
      /**
       * 菜单项的默认值
       */
      get data() {
        if (__PopsPanel__.data == null) {
          __PopsPanel__.data = new utils.Dictionary();
        }
        return __PopsPanel__.data;
      },
      /**
       * 成功只执行了一次的项
       */
      get oneSuccessExecMenu() {
        if (__PopsPanel__.oneSuccessExecMenu == null) {
          __PopsPanel__.oneSuccessExecMenu = new utils.Dictionary();
        }
        return __PopsPanel__.oneSuccessExecMenu;
      },
      /**
       * 成功只执行了一次的项
       */
      get onceExec() {
        if (__PopsPanel__.onceExec == null) {
          __PopsPanel__.onceExec = new utils.Dictionary();
        }
        return __PopsPanel__.onceExec;
      },
      /** 脚本名,一般用在设置的标题上 */
      get scriptName() {
        return SCRIPT_NAME;
      },
      /** 菜单项的总值在本地数据配置的键名 */
      key: KEY,
      /** 菜单项在attributes上配置的菜单键 */
      attributeKeyName: ATTRIBUTE_KEY,
      /** 菜单项在attributes上配置的菜单默认值 */
      attributeDefaultValueName: ATTRIBUTE_DEFAULT_VALUE
    },
    /** 监听器 */
    $listener: {
      /**
       * 值改变的监听器
       */
      get listenData() {
        if (__PopsPanel__.listenData == null) {
          __PopsPanel__.listenData = new utils.Dictionary();
        }
        return __PopsPanel__.listenData;
      }
    },
    init() {
      this.initPanelDefaultValue();
      this.initExtensionsMenu();
    },
    initExtensionsMenu() {
      if (_unsafeWindow.top !== _unsafeWindow.self) {
        return;
      }
      GM_Menu.add([
        {
          key: "show_pops_panel_setting",
          text: i18next.t("⚙ 设置"),
          autoReload: false,
          isStoreValue: false,
          showText(text) {
            return text;
          },
          callback: () => {
            this.showPanel();
          }
        }
      ]);
    },
    /** 初始化本地设置默认的值 */
    initPanelDefaultValue() {
      let that = this;
      function initDefaultValue(config) {
        if (!config["attributes"]) {
          return;
        }
        let key = config.attributes[ATTRIBUTE_KEY];
        let defaultValue = config["attributes"][ATTRIBUTE_DEFAULT_VALUE];
        if (key == null) {
          log.warn(["请先配置键", config]);
          return;
        }
        if (that.$data.data.has(key)) {
          log.warn("请检查该key(已存在): " + key);
        }
        that.$data.data.set(key, defaultValue);
      }
      function loopInitDefaultValue(configList) {
        for (let index = 0; index < configList.length; index++) {
          let configItem = configList[index];
          initDefaultValue(configItem);
          let childForms = configItem.forms;
          if (childForms && Array.isArray(childForms)) {
            loopInitDefaultValue(childForms);
          }
        }
      }
      let contentConfigList = this.getPanelContentConfig();
      for (let index = 0; index < contentConfigList.length; index++) {
        let leftContentConfigItem = contentConfigList[index];
        if (!leftContentConfigItem.forms) {
          continue;
        }
        let rightContentConfigList = leftContentConfigItem.forms;
        if (rightContentConfigList && Array.isArray(rightContentConfigList)) {
          loopInitDefaultValue(rightContentConfigList);
        }
      }
    },
    /**
     * 设置值
     * @param key 键
     * @param value 值
     */
    setValue(key, value) {
      let locaData = _GM_getValue(KEY, {});
      let oldValue = locaData[key];
      locaData[key] = value;
      _GM_setValue(KEY, locaData);
      if (this.$listener.listenData.has(key)) {
        this.$listener.listenData.get(key).callback(key, oldValue, value);
      }
    },
    /**
     * 获取值
     * @param key 键
     * @param defaultValue 默认值
     */
    getValue(key, defaultValue) {
      let locaData = _GM_getValue(KEY, {});
      let localValue = locaData[key];
      if (localValue == null) {
        if (this.$data.data.has(key)) {
          return this.$data.data.get(key);
        }
        return defaultValue;
      }
      return localValue;
    },
    /**
     * 删除值
     * @param key 键
     */
    deleteValue(key) {
      let locaData = _GM_getValue(KEY, {});
      let oldValue = locaData[key];
      Reflect.deleteProperty(locaData, key);
      _GM_setValue(KEY, locaData);
      if (this.$listener.listenData.has(key)) {
        this.$listener.listenData.get(key).callback(key, oldValue, void 0);
      }
    },
    /**
     * 监听调用setValue、deleteValue
     * @param key 需要监听的键
     * @param callback
     */
    addValueChangeListener(key, callback, option) {
      let listenerId = Math.random();
      this.$listener.listenData.set(key, {
        id: listenerId,
        key,
        callback
      });
      if (option) {
        if (option.immediate) {
          callback(key, this.getValue(key), this.getValue(key));
        }
      }
      return listenerId;
    },
    /**
     * 移除监听
     * @param listenerId 监听的id
     */
    removeValueChangeListener(listenerId) {
      let deleteKey = null;
      for (const [key, value] of this.$listener.listenData.entries()) {
        if (value.id === listenerId) {
          deleteKey = key;
          break;
        }
      }
      if (typeof deleteKey === "string") {
        this.$listener.listenData.delete(deleteKey);
      } else {
        console.warn("没有找到对应的监听器");
      }
    },
    /**
     * 判断该键是否存在
     * @param key 键
     */
    hasKey(key) {
      let locaData = _GM_getValue(KEY, {});
      return key in locaData;
    },
    /**
     * 自动判断菜单是否启用,然后执行回调
     * @param key
     * @param callback 回调
     */
    execMenu(key, callback) {
      if (typeof key !== "string") {
        throw new TypeError("key 必须是字符串");
      }
      if (!this.$data.data.has(key)) {
        log.warn(`${key} 键不存在`);
        return;
      }
      let value = PopsPanel.getValue(key);
      if (value) {
        callback(value);
      }
    },
    /**
     * 自动判断菜单是否启用,然后执行回调,只会执行一次
     * @param key
     * @param callback 回调
     */
    execMenuOnce(key, callback) {
      if (typeof key !== "string") {
        throw new TypeError("key 必须是字符串");
      }
      if (!this.$data.data.has(key)) {
        log.warn(`${key} 键不存在`);
        return;
      }
      if (this.$data.oneSuccessExecMenu.has(key)) {
        return;
      }
      this.$data.oneSuccessExecMenu.set(key, 1);
      let resultStyleList = [];
      let pushStyleNode = (style) => {
        let __value = PopsPanel.getValue(key);
        changeCallBack(__value, style);
      };
      let changeCallBack = (currentValue, resultStyle) => {
        let resultList = [];
        if (currentValue) {
          let result = resultStyle ?? callback(currentValue, pushStyleNode);
          if (result instanceof HTMLStyleElement) {
            resultList = [result];
          } else if (Array.isArray(result)) {
            resultList = [
              ...result.filter(
                (item) => item != null && item instanceof HTMLStyleElement
              )
            ];
          }
        }
        for (let index = 0; index < resultStyleList.length; index++) {
          let $css = resultStyleList[index];
          $css.remove();
          resultStyleList.splice(index, 1);
          index--;
        }
        resultStyleList = [...resultList];
      };
      this.addValueChangeListener(
        key,
        (__key, oldValue, newValue) => {
          changeCallBack(newValue);
        }
      );
      let value = PopsPanel.getValue(key);
      if (value) {
        changeCallBack(value);
      }
    },
    /**
     * 根据key执行一次
     * @param key
     */
    onceExec(key, callback) {
      if (typeof key !== "string") {
        throw new TypeError("key 必须是字符串");
      }
      if (this.$data.onceExec.has(key)) {
        return;
      }
      callback();
      this.$data.onceExec.set(key, 1);
    },
    /**
     * 显示设置面板
     */
    showPanel() {
      __pops.panel({
        title: {
          text: i18next.t("{{SCRIPT_NAME}}-设置", { SCRIPT_NAME }),
          position: "center",
          html: false,
          style: ""
        },
        content: this.getPanelContentConfig(),
        mask: {
          enable: true,
          clickEvent: {
            toClose: true,
            toHide: false
          }
        },
        isMobile: this.isMobile(),
        width: this.getWidth(),
        height: this.getHeight(),
        drag: true,
        only: true,
        style: `
			${UIScriptListCSS}
			`
      });
    },
    isMobile() {
      return window.outerWidth < 550;
    },
    /**
     * 获取设置面板的宽度
     */
    getWidth() {
      if (window.outerWidth < 550) {
        return "92vw";
      } else {
        return "550px";
      }
    },
    /**
     * 获取设置面板的高度
     */
    getHeight() {
      if (window.outerHeight > 450) {
        return "80vh";
      } else {
        return "450px";
      }
    },
    /**
     * 获取配置内容
     */
    getPanelContentConfig() {
      let configList = [
        SettingUIGeneral,
        SettingUIScripts,
        SettingUIDiscuessions,
        SettingUIUsers,
        SettingUIScriptList,
        SettingUIScriptLib
      ];
      return configList;
    }
  };
  PopsPanel.init();
  Greasyfork.init();

})(Qmsg, DOMUtils, Utils, i18next, pops, Viewer);