// ==UserScript==
// @name GreasyFork优化
// @name:en-US GreasyFork Optimization
// @namespace https://github.com/WhiteSevs/TamperMonkeyScript
// @version 2024.7.17.20
// @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.icu/scripts/494167/1376186/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.6.1/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.2.4/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.1/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
// @downloadURL none
// ==/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脚本,🖐点我安装': '需安装Greasyfork Beautify脚本,🖐点我安装',
代码: "代码",
添加复制代码按钮: "添加复制代码按钮",
更优雅的复制: "更优雅的复制",
快捷键: "快捷键",
"【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秒后更新下一个",
全部更新失败: "全部更新失败",
"全部更新完毕
成功:{{successNums}}
失败:{{failedNums}}
总计:{{scriptUrlListLength}}": "全部更新完毕
成功:{{successNums}}
失败:{{failedNums}}
总计:{{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}} 版本"
};
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脚本,🖐点我安装': 'Greasyfork Beauty script needs to be installed,🖐 Click me to install',
代码: "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",
"全部更新完毕
成功:{{successNums}}
失败:{{failedNums}}
总计:{{scriptUrlListLength}}": "All updates completed
Success: {{successNums}}
Failure: {{failed Nums}}
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"
};
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");
},
/**
* 讨论页面
*/
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 `
${i18next.t("自定义已读颜色")}
${i18next.t("在讨论内生效")}
` }); let $right = domUtils.createElement("div", { className: "pops-panel-item-right", innerHTML: ` ` }); 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: "", type: "forms", forms: [ { text: i18next.t("过滤"), type: "deepMenu", forms: [ { text: "", 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)") ) ] }, { text: "", type: "forms", forms: [ { text: i18next.t("过滤脚本(id)"), type: "deepMenu", forms: [ { text: "", type: "forms", forms: [ { type: "own", getLiElementCallBack(liElement) { let textareaDiv = domUtils.createElement( "div", { className: "pops-panel-textarea", innerHTML: ` ` }, { style: "width: 100%;" } ); let textarea = textareaDiv.querySelector( "textarea" ); const KEY2 = "greasyfork-discussions-filter-script"; textarea.value = PopsPanel.getValue(KEY2, ""); domUtils.on( textarea, ["input", "propertychange"], void 0, utils.debounce(function(event) { PopsPanel.setValue(KEY2, textarea.value); }, 200) ); liElement.appendChild(textareaDiv); return liElement; } } ] } ] }, { text: i18next.t("过滤发布的用户(id)"), type: "deepMenu", forms: [ { text: "", type: "forms", forms: [ { type: "own", getLiElementCallBack(liElement) { let textareaDiv = domUtils.createElement( "div", { className: "pops-panel-textarea", innerHTML: ` ` }, { style: "width: 100%;" } ); let textarea = textareaDiv.querySelector( "textarea" ); const KEY2 = "greasyfork-discussions-filter-post-user"; textarea.value = PopsPanel.getValue(KEY2, ""); domUtils.on( textarea, ["input", "propertychange"], void 0, utils.debounce(function(event) { PopsPanel.setValue(KEY2, textarea.value); }, 200) ); liElement.appendChild(textareaDiv); return liElement; } } ] } ] }, { text: i18next.t("过滤回复的用户(id)"), type: "deepMenu", forms: [ { text: "", type: "forms", forms: [ { type: "own", getLiElementCallBack(liElement) { let textareaDiv = domUtils.createElement( "div", { className: "pops-panel-textarea", innerHTML: ` ` }, { style: "width: 100%;" } ); let textarea = textareaDiv.querySelector( "textarea" ); const KEY2 = "greasyfork-discussions-filter-reply-user"; textarea.value = PopsPanel.getValue(KEY2, ""); domUtils.on( textarea, ["input", "propertychange"], void 0, utils.debounce(function(event) { PopsPanel.setValue(KEY2, 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 -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 -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 beautifyTextAreaCSS = "textarea {\r\n position: relative;\r\n display: inline-block;\r\n width: 100%;\r\n vertical-align: bottom;\r\n font-size: 14px;\r\n position: relative;\r\n display: block;\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 -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 }\r\n header#main-header + div {\r\n margin-top: calc(var(--header-height) + 35px);\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 .width-constraint {\r\n display: flex;\r\n align-items: center;\r\n gap: 20px;\r\n padding: unset;\r\n }\r\n\r\n #site-name {\r\n display: flex;\r\n img {\r\n width: calc(var(--header-height) - 5px);\r\n height: calc(var(--header-height) - 5px);\r\n }\r\n }\r\n\r\n /* 隐藏Greasyfork文字 */\r\n #site-name-text {\r\n display: none;\r\n }\r\n\r\n #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 nav a {\r\n text-decoration: none;\r\n }\r\n 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 nav li:hover {\r\n background: #5f0101;\r\n }\r\n }\r\n\r\n #nav-user-info {\r\n max-width: 150px;\r\n .user-profile-link {\r\n /*overflow: hidden;\r\n white-space: nowrap;\r\n text-overflow: ellipsis;*/\r\n }\r\n > span {\r\n flex: 1;\r\n }\r\n }\r\n #nav-user-info,\r\n #nav-user-info + nav {\r\n position: unset;\r\n width: unset;\r\n /* height: 100%; */\r\n display: flex;\r\n flex-wrap: wrap;\r\n align-items: center;\r\n }\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(beautifyTextAreaCSS)); result.push( _GM_addStyle(` 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(` .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(` 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(` 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 GreasyforkDiscussionsFilter = { $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(` .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.filterDiscussions(); }, 50); utils.mutationObserver(document.body, { config: { subtree: true, childList: true }, callback: () => { lockFunction.run(); } }); lockFunction.run(); }, /** * 论坛-过滤 */ filterDiscussions() { const filterScript = PopsPanel.getValue(this.$data.FILTER_SCRIPT_KEY, ""); const filterPostUser = PopsPanel.getValue( this.$data.FILTER_POST_USER_KEY, "" ); const filterReplyUser = PopsPanel.getValue( this.$data.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"); const SNIPPET_MAP = /* @__PURE__ */ new Map(); document.querySelectorAll(".discussion-list-container").forEach(($listContainer, index) => { if (!$listContainer.querySelector("a.script-link")) { return; } const discussionInfo = this.parseDiscussionListContainer($listContainer); 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 (const filterScriptId of filterScriptList) { if (discussionInfo.scriptId === filterScriptId) { log.success([ `过滤脚本id:${discussionInfo.scriptId}`, discussionInfo ]); $listContainer.remove(); return; } } for (const filterPostUserId of filterPostUserList) { if (discussionInfo.postUserId === filterPostUserId) { log.success([ `过滤发布用户id:${discussionInfo.postUserId}`, discussionInfo ]); $listContainer.remove(); return; } } if (discussionInfo.replyUserName) { for (const filterReplyUserId of filterReplyUserList) { if (discussionInfo.replyUserId === filterReplyUserId) { log.success([ `过滤回复用户id:${discussionInfo.replyUserId}`, discussionInfo ]); $listContainer.remove(); return; } } } }); }, /** * 解析出元素上的属性 */ parseDiscussionListContainer($listContainer) { var _a2; const discussionInfo = { /** 脚本名 */ scriptName: $listContainer.querySelector("a.script-link").innerText, /** 脚本主页地址 */ scriptUrl: $listContainer.querySelector("a.script-link").href, /** 脚本id */ scriptId: GreasyforkApi.getScriptId( $listContainer.querySelector("a.script-link").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: $listContainer.querySelector( "span.discussion-snippet" ).innerText, /** 回复的用户名*/ replyUserName: void 0, /** 回复的用户主页地址*/ replyUserHomeUrl: void 0, /** 回复的用户id*/ replyUserId: void 0, /** 回复的时间 */ replyTimeStamp: void 0 }; if ($listContainer.querySelector( ".discussion-meta-item .discussion-meta-item" )) { discussionInfo.replyUserName = $listContainer.querySelector( ".discussion-meta-item .discussion-meta-item a.user-link" ).innerText; discussionInfo.replyUserHomeUrl = $listContainer.querySelector( ".discussion-meta-item .discussion-meta-item a.user-link" ).href; discussionInfo.replyUserId = GreasyforkApi.getUserId( discussionInfo.replyUserHomeUrl ); discussionInfo.replyTimeStamp = new Date( (_a2 = $listContainer.querySelector( ".discussion-meta-item .discussion-meta-item relative-time" )) == null ? void 0 : _a2.getAttribute("datetime") ); } return discussionInfo; } }; 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(` .discussion-read{ background: ${color} !important; } `); }, /** * 添加快捷操作按钮 */ addShortcutOperationButton() { log.info("添加快捷操作按钮"); document.querySelectorAll(".discussion-list-container").forEach(($listContainer) => { if (!$listContainer.querySelector("a.script-link")) { return; } let $listItem = $listContainer.querySelector( ".discussion-list-item" ); let $meta = $listItem.querySelector(".discussion-meta"); let $ownMetaItem = domUtils.createElement("div", { className: "discussion-meta-item", innerHTML: ` ` }); let $button = $ownMetaItem.querySelector( ".discussion-filter-button" ); $meta.appendChild($ownMetaItem); domUtils.on($button, "click", (event) => { utils.preventEvent(event); const discussionInfo = GreasyforkDiscussionsFilter.parseDiscussionListContainer( $listContainer ); let $popsDialog = __pops.alert({ title: { text: i18next.t("选择需要过滤的选项"), position: "center", html: false }, content: { text: `${i18next.t("评分:")}${scriptInfo["fan_score"]}
${i18next.t("语言:")}${scriptInfo["locale"]}
${i18next.t("版本:")}${scriptInfo["version"]}
${i18next.t("更新:")}${utils.getDaysDifference( new Date(scriptInfo["code_updated_at"]).getTime(), void 0, "auto" )}前
${i18next.t("同步方式:{{syncMode}}", { syncMode })}