GreasyFork优化

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

当前为 2024-12-10 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name GreasyFork优化
  3. // @name:en-US GreasyFork Optimization
  4. // @namespace https://github.com/WhiteSevs/TamperMonkeyScript
  5. // @version 2024.12.10
  6. // @author WhiteSevs
  7. // @description 自动登录账号、快捷寻找自己库被其他脚本引用、更新自己的脚本列表、库、优化图片浏览、美化页面、Markdown复制按钮
  8. // @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
  9. // @license GPL-3.0-only
  10. // @icon 
  11. // @supportURL https://github.com/WhiteSevs/TamperMonkeyScript/issues
  12. // @match *://greasyfork.org/*
  13. // @require https://update.greasyfork.org/scripts/494167/1413255/CoverUMD.js
  14. // @require https://fastly.jsdelivr.net/npm/@whitesev/utils@2.5.4/dist/index.umd.js
  15. // @require https://fastly.jsdelivr.net/npm/@whitesev/domutils@1.4.8/dist/index.umd.js
  16. // @require https://fastly.jsdelivr.net/npm/@whitesev/pops@1.9.5/dist/index.umd.js
  17. // @require https://fastly.jsdelivr.net/npm/qmsg@1.2.8/dist/index.umd.js
  18. // @require https://fastly.jsdelivr.net/npm/viewerjs@1.11.6/dist/viewer.min.js
  19. // @require https://fastly.jsdelivr.net/npm/i18next@23.15.1/i18next.min.js
  20. // @resource ViewerCSS https://fastly.jsdelivr.net/npm/viewerjs@1.11.6/dist/viewer.min.css
  21. // @connect greasyfork.org
  22. // @grant GM_addStyle
  23. // @grant GM_deleteValue
  24. // @grant GM_getResourceText
  25. // @grant GM_getValue
  26. // @grant GM_info
  27. // @grant GM_registerMenuCommand
  28. // @grant GM_setValue
  29. // @grant GM_unregisterMenuCommand
  30. // @grant GM_xmlhttpRequest
  31. // @grant unsafeWindow
  32. // @run-at document-start
  33. // ==/UserScript==
  34.  
  35. (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} ");
  36.  
  37. (function (Qmsg, DOMUtils, Utils, i18next, pops, Viewer) {
  38. 'use strict';
  39.  
  40. var __defProp = Object.defineProperty;
  41. var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
  42. var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
  43. var _a;
  44. var _GM_deleteValue = /* @__PURE__ */ (() => typeof GM_deleteValue != "undefined" ? GM_deleteValue : void 0)();
  45. var _GM_getResourceText = /* @__PURE__ */ (() => typeof GM_getResourceText != "undefined" ? GM_getResourceText : void 0)();
  46. var _GM_getValue = /* @__PURE__ */ (() => typeof GM_getValue != "undefined" ? GM_getValue : void 0)();
  47. var _GM_info = /* @__PURE__ */ (() => typeof GM_info != "undefined" ? GM_info : void 0)();
  48. var _GM_registerMenuCommand = /* @__PURE__ */ (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : void 0)();
  49. var _GM_setValue = /* @__PURE__ */ (() => typeof GM_setValue != "undefined" ? GM_setValue : void 0)();
  50. var _GM_unregisterMenuCommand = /* @__PURE__ */ (() => typeof GM_unregisterMenuCommand != "undefined" ? GM_unregisterMenuCommand : void 0)();
  51. var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)();
  52. var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)();
  53. var _monkeyWindow = /* @__PURE__ */ (() => window)();
  54. const zh_CN_language = {
  55. GreasyFork优化: "GreasyFork优化",
  56. 请求取消: "请求取消",
  57. 请求超时: "请求超时",
  58. 请求异常: "请求异常",
  59. 通用: "通用",
  60. 账号: "账号",
  61. 密码: "密码",
  62. 语言: "语言",
  63. "账号/密码": "账号/密码",
  64. 请输入账号: "请输入账号",
  65. 请输入密码: "请输入密码",
  66. 自动登录: "自动登录",
  67. 自动登录当前保存的账号: "自动登录当前保存的账号",
  68. "清空账号/密码": "清空账号/密码",
  69. 点击清空: "点击清空",
  70. "确定清空账号和密码?": "确定清空账号和密码?",
  71. "已清空账号/密码": "已清空账号/密码",
  72. "源代码同步【脚本列表】": "源代码同步【脚本列表】",
  73. 一键同步: "一键同步",
  74. 前往用户主页: "前往用户主页",
  75. 获取当前已登录的用户主页失败: "获取当前已登录的用户主页失败",
  76. "源代码同步【未上架的脚本】": "源代码同步【未上架的脚本】",
  77. "源代码同步【库】": "源代码同步【库】",
  78. 论坛: "论坛",
  79. 功能: "功能",
  80. 过滤重复的评论: "过滤重复的评论",
  81. "过滤掉重复的评论数量(≥2)": "过滤掉重复的评论数量(≥2)",
  82. "过滤脚本(id)": "过滤脚本(id)",
  83. "请输入脚本id,每行一个": "请输入脚本id,每行一个",
  84. "过滤发布的用户(id)": "过滤发布的用户(id)",
  85. "请输入用户id,每行一个": "请输入用户id,每行一个",
  86. "过滤回复的用户(id)": "过滤回复的用户(id)",
  87. 优化: "优化",
  88. 固定当前语言: "固定当前语言",
  89. 无: "无",
  90. "如button、input、textarea": "如button、input、textarea",
  91. 更直观的查看版本迭代: "更直观的查看版本迭代",
  92. 美化上传图片按钮: "美化上传图片按钮",
  93. 放大上传区域: "放大上传区域",
  94. 优化图片浏览: "优化图片浏览",
  95. 使用Viewer浏览图片: "使用Viewer浏览图片",
  96. 覆盖图床图片跳转: "覆盖图床图片跳转",
  97. "配合上面的【优化图片浏览】更优雅浏览图片": "配合上面的【优化图片浏览】更优雅浏览图片",
  98. '需安装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>',
  99. 代码: "代码",
  100. 添加复制代码按钮: "添加复制代码按钮",
  101. 更优雅的复制: "更优雅的复制",
  102. 快捷键: "快捷键",
  103. "【F】键全屏、【Alt+Shift+F】键宽屏": "【F】键全屏、【Alt+Shift+F】键宽屏",
  104. 库: "库",
  105. 脚本列表: "脚本列表",
  106. "请输入屏蔽规则,每行一个": "请输入屏蔽规则,每行一个",
  107. 请求admin内容失败: "请求admin内容失败",
  108. 解析admin的源代码同步表单失败: "解析admin的源代码同步表单失败",
  109. 源代码同步失败: "源代码同步失败",
  110. 获取用户信息失败: "获取用户信息失败",
  111. 获取用户的收藏集失败: "获取用户的收藏集失败",
  112. "解析Script Sets失败": "解析Script Sets失败",
  113. "获取收藏集{{setsId}}失败": "获取收藏集{{setsId}}失败",
  114. "获取表单元素#edit_script_set失败": "获取表单元素#edit_script_set失败",
  115. 更新收藏集表单请求失败: "更新收藏集表单请求失败",
  116. 请先在菜单中录入账号: "请先在菜单中录入账号",
  117. 请先在菜单中录入密码: "请先在菜单中录入密码",
  118. "获取csrf-token失败": "获取csrf-token失败",
  119. "正在登录中...": "正在登录中...",
  120. "登录失败,请在控制台查看原因": "登录失败,请在控制台查看原因",
  121. "登录成功,1s后自动跳转": "登录成功,1s后自动跳转",
  122. "登录失败,可能是账号/密码错误,请在控制台查看原因": "登录失败,可能是账号/密码错误,请在控制台查看原因",
  123. "美化 历史版本 页面": "美化 历史版本 页面",
  124. 未找到history_versions元素列表: "未找到history_versions元素列表",
  125. "yyyy年MM月dd日 HH:mm:ss": "yyyy-MM-dd HH:mm:ss",
  126. "美化 Greasyfork Beautify脚本": "美化 Greasyfork Beautify脚本",
  127. "❌ 最多同时长传5张图": "❌ 最多同时长传5张图片",
  128. "❌ 图片:{{name}} 大小:{{size}}": "❌ 图片:{{name}} 大小:{{size}}",
  129. "已过滤:{{oldCount}}": "已过滤:{{oldCount}}",
  130. 寻找引用: "寻找引用",
  131. 获取脚本id失败: "获取脚本id失败",
  132. 收藏: "收藏",
  133. 请先登录账号: "请先登录账号",
  134. 获取用户id失败: "获取用户id失败",
  135. "获取收藏夹中...": "获取收藏夹中...",
  136. 收藏集: "收藏集",
  137. "添加中...": "添加中...",
  138. "添加失败,{{selector}}元素不存在": "添加失败,{{selector}}元素不存在",
  139. "未找到{{selector}}元素": "未找到{{selector}}元素",
  140. 添加失败: "添加失败",
  141. 添加成功: "添加成功",
  142. "删除中...": "删除中...",
  143. 删除成功: "删除成功",
  144. 添加: "添加",
  145. 刪除: "刪除",
  146. "拦截跳转:": "拦截跳转:",
  147. 今日检查: "今日检查",
  148. 复制代码: "复制代码",
  149. "加载文件中...": "加载文件中...",
  150. 复制成功: "复制成功",
  151. "✅ 复制成功!": "✅ 复制成功!",
  152. "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}": "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}",
  153. "导航至:": "导航至:",
  154. "请先登录账号!": "请先登录账号!",
  155. "获取信息中,请稍后...": "获取信息中,请稍后...",
  156. "获取成功,共 {{count}} 个": "获取成功,共 {{count}} 个",
  157. "评分:": "评分:",
  158. "语言:": "语言:",
  159. "版本:": "版本:",
  160. "更新:": "更新:",
  161. 同步代码: "同步代码",
  162. "同步中...": "同步中...",
  163. 手动: "手动",
  164. 自动: "自动",
  165. "同步方式:{{syncMode}}": "同步方式:{{syncMode}}",
  166. 同步成功: "同步成功",
  167. 同步失败: "同步失败",
  168. 该脚本未设置同步信息: "该脚本未设置同步信息",
  169. "上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载": "上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载",
  170. "名称:": "名称:",
  171. "进度:": "进度:",
  172. "未获取到【脚本列表】": "未获取到【脚本列表】",
  173. "源代码同步成功,3秒后更新下一个": "源代码同步成功,3秒后更新下一个",
  174. 全部更新失败: "全部更新失败",
  175. "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}": "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}",
  176. "⚙ 设置": "⚙ 设置",
  177. "{{SCRIPT_NAME}}-设置": "{{SCRIPT_NAME}}-设置",
  178. 美化页面元素: "美化页面元素",
  179. 美化历史版本页面: "美化历史版本页面",
  180. "美化Greasyfork Beautify脚本": "美化Greasyfork Beautify脚本",
  181. 获取表单csrfToken失败: "获取表单csrfToken失败",
  182. Toast配置: "Toast配置",
  183. Toast位置: "Toast位置",
  184. 左上角: "左上角",
  185. 顶部: "顶部",
  186. 右上角: "右上角",
  187. 左边: "左边",
  188. 中间: "中间",
  189. 右边: "右边",
  190. 左下角: "左下角",
  191. 底部: "底部",
  192. 右下角: "右下角",
  193. Toast显示在页面九宫格的位置: "Toast显示在页面九宫格的位置",
  194. 最多显示的数量: "最多显示的数量",
  195. 限制Toast显示的数量: "限制Toast显示的数量",
  196. 逆序弹出: "逆序弹出",
  197. 修改Toast弹出的顺序: "修改Toast弹出的顺序",
  198. 该脚本已经在该收藏集中: "该脚本已经在该收藏集中",
  199. 其它错误: "其它错误",
  200. 启用: "启用",
  201. 开启后下面的过滤功能才会生效: "开启后下面的功能才会生效",
  202. 屏蔽脚本: "屏蔽脚本",
  203. 点击查看规则: "点击查看规则",
  204. 过滤: "过滤",
  205. 代码同步: "代码同步",
  206. 美化: "美化",
  207. 修复代码行号显示: "修复代码行号显示",
  208. 修复代码行数超过1k行号显示不全问题: "修复代码行数超过1k行号显示不全问题",
  209. "添加【寻找引用】按钮": "添加【寻找引用】按钮",
  210. "在脚本栏添加按钮,一般用于搜索引用该库的相关脚本": "在脚本栏添加按钮,一般用于搜索引用该库的相关脚本",
  211. "添加【收藏】按钮": "添加【收藏】按钮",
  212. "在脚本栏添加按钮,一般用于快捷收藏该脚本/库": "在脚本栏添加按钮,一般用于快捷收藏该脚本/库",
  213. 修复图片宽度显示问题: "修复图片宽度显示问题",
  214. 修复图片在移动端宽度超出浏览器宽度问题: "修复图片在移动端宽度超出浏览器宽度问题",
  215. "添加【今日检查】信息块": "添加【今日检查】信息块",
  216. "在脚本信息栏添加【今日检查】信息块": "在脚本信息栏添加【今日检查】信息块",
  217. "给Markdown添加【复制】按钮": "给Markdown添加【复制】按钮",
  218. "在Markdown内容右上角添加【复制】按钮,点击一键复制Markdown内容": "在Markdown内容右上角添加【复制】按钮,点击一键复制Markdown内容",
  219. 开启后下面的功能才会生效: "开启后下面的功能才会生效",
  220. 检测页面加载: "检测页面加载",
  221. "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面": "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面",
  222. 检测间隔: "检测间隔",
  223. "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面": "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面",
  224. 美化顶部导航栏: "美化顶部导航栏",
  225. "可能会跟Greasyfork Beautify脚本有冲突": "可能会跟Greasyfork Beautify脚本有冲突",
  226. 美化脚本列表: "美化脚本列表",
  227. "双列显示且添加脚本卡片操作项(安装、收藏)": "双列显示且添加脚本卡片操作项(安装、收藏)",
  228. 操作面板: "操作面板",
  229. "添加【操作面板】按钮": "添加【操作面板】按钮",
  230. "在脚本列表页面时为顶部导航栏添加【操作面板】按钮": "在脚本列表页面时为顶部导航栏添加【操作面板】按钮",
  231. 操作: "操作",
  232. 安装此脚本: "安装此脚本",
  233. 脚本: "脚本",
  234. 历史版本: "历史版本",
  235. 自定义已读颜色: "自定义已读颜色",
  236. 在讨论内生效: "在讨论内生效",
  237. 用户: "用户",
  238. 控制台: "控制台",
  239. "迁移【控制台】到顶部导航栏": "迁移【控制台】到顶部导航栏",
  240. "将【控制台】按钮移动到顶部导航栏,节省空间": "将【控制台】按钮移动到顶部导航栏,节省空间",
  241. "在版本下面添加【安装】、【查看代码】按钮": "在版本下面添加【安装】、【查看代码】按钮",
  242. 查看代码: "查看代码",
  243. 添加快捷操作按钮: "添加快捷操作按钮",
  244. "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效": "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效",
  245. 选择需要过滤的选项: "选择需要过滤的选项",
  246. "确定{{type}}:{{filterId}}?": "确定{{type}}:{{filterId}}?",
  247. "已删除:{{scriptId}}": "已删除:{{scriptId}}",
  248. 帮助文档: "帮助文档",
  249. "请输入规则,每行一个": "请输入规则,每行一个",
  250. 选择过滤的选项: "选择过滤的选项",
  251. "脚本id:{{text}}": "脚本id:{{text}}",
  252. "脚本名:{{text}}": "脚本名:{{text}}",
  253. "作者id:{{text}}": "作者id:{{text}}",
  254. "作者名:{{text}}": "作者名:{{text}}",
  255. "作用域:脚本、脚本搜索、用户主页": "作用域:脚本、脚本搜索、用户主页",
  256. "更新到 {{version}} 版本": "更新到 {{version}} 版本",
  257. "降级到 {{version}} 版本": "降级到 {{version}} 版本",
  258. "重新安装 {{version}} 版本": "重新安装 {{version}} 版本",
  259. "发布的用户id:{{text}}": "发布的用户id:{{text}}",
  260. 自定义快捷键: "自定义快捷键",
  261. 点击录入快捷键: "点击录入快捷键",
  262. 快捷键发表回复: "快捷键发表回复",
  263. "在输入框内按下快捷发表回复,例如:{{key}}": "在输入框内按下快捷发表回复,例如:{{key}}",
  264. 请先执行当前的录入操作: "请先执行当前的录入操作",
  265. 清空快捷键: "清空快捷键",
  266. "请按下快捷键...": "请按下快捷键...",
  267. 成功录入: "成功录入",
  268. "快捷键 {{key}} 已被 {{isUsedKey}} 占用": "快捷键 {{key}} 已被 {{isUsedKey}} 占用",
  269. 私聊: "私聊",
  270. 美化私信页面: "美化私信页面",
  271. 美化为左右对话模式: "美化为左右对话模式",
  272. "最后回复:": "最后回复:",
  273. 进入: "进入",
  274. 记住回复内容: "记住回复内容",
  275. "监听表单内的textarea内容改变并存储到indexDB中,提交表单将清除保存的数据,误刷新页面时可动态恢复": "监听表单内的textarea内容改变并存储到indexDB中,提交表单将清除保存的数据,误刷新页面时可动态恢复",
  276. 表单: "表单",
  277. 自动清理空间: "自动清理空间",
  278. 不清理: "不清理",
  279. "{{value}} 天": "{{value}} 天",
  280. "{{value}} 周": "{{value}} 周",
  281. "{{value}} 个月": "{{value}} 个月",
  282. 半年: "半年",
  283. 计算中: "计算中",
  284. 根据设置的间隔时间自动清理保存的回复内容: "根据设置的间隔时间自动清理保存的回复内容",
  285. "数据占用空间:{{size}}": "数据占用空间:{{size}}",
  286. 当前存储的数据所占用的空间大小: "当前存储的数据所占用的空间大小",
  287. 清空: "清空",
  288. 清理成功: "清理成功",
  289. 清理失败: "清理失败",
  290. "Url To WebhookUrl": "Url 转 WebhookUrl",
  291. 关闭: "关闭",
  292. "例如:": "例如:",
  293. "结果:": "结果:",
  294. 转换前: "转换前",
  295. 转换后: "转换后",
  296. 使用namespace查询脚本信息: "使用namespace查询脚本信息",
  297. 脚本管理: "脚本管理",
  298. "开启后检测已安装的脚本信息更准确,但是速度会更慢": "开启后检测已安装的脚本信息更准确,但是速度会更慢",
  299. 美化私信列表: "美化私信列表",
  300. 搜索: "搜索"
  301. };
  302. const en_US_language = {
  303. GreasyFork优化: "GreasyFork Optimization",
  304. 请求取消: "http request cancel",
  305. 请求超时: "http request timeout",
  306. 请求异常: "http request error",
  307. 通用: "General",
  308. 账号: "Account",
  309. 密码: "Password",
  310. 语言: "Language",
  311. "账号/密码": "Account/Password",
  312. 请输入账号: "Please enter your account number",
  313. 请输入密码: "Please enter password",
  314. 自动登录: "Auto Login",
  315. 自动登录当前保存的账号: "Automatically log in to the currently saved account",
  316. "清空账号/密码": "Clear account/password",
  317. 点击清空: "Clear",
  318. "确定清空账号和密码?": "Are you sure to clear your account and password?",
  319. "已清空账号/密码": "Account/password cleared",
  320. "源代码同步【脚本列表】": "Source Code Synchronization [Script List]",
  321. 一键同步: "Sync All",
  322. 前往用户主页: "Go to the user's homepage",
  323. 获取当前已登录的用户主页失败: "Failed to retrieve the currently logged in user's homepage",
  324. "源代码同步【未上架的脚本】": "Source code synchronization [Script not listed]",
  325. "源代码同步【库】": "Source code synchronization [Library]",
  326. 论坛: "Forum",
  327. 功能: "Features",
  328. 过滤重复的评论: "Filter duplicate comments",
  329. "过滤掉重复的评论数量(≥2)": "Filter out duplicate comments (≥ 2)",
  330. "过滤脚本(id)": "Filter script (id)",
  331. "请输入脚本id,每行一个": "Please enter the script ID, one per line",
  332. "过滤发布的用户(id)": "Filter published users (id)",
  333. "请输入用户id,每行一个": "Please enter the user ID, one per line",
  334. "过滤回复的用户(id)": "User (ID) who filters replies",
  335. 优化: "Optimization",
  336. 固定当前语言: "Fix current language",
  337. 无: "nothing",
  338. "如button、input、textarea": "For example button、input、textarea",
  339. 更直观的查看版本迭代: "More intuitive viewing of version iterations",
  340. 美化上传图片按钮: "Beautify upload image button",
  341. 放大上传区域: "Enlarge the upload area",
  342. 优化图片浏览: "Optimize image browsing",
  343. 使用Viewer浏览图片: "Using Viewer to browse images",
  344. 覆盖图床图片跳转: "Overlay bed image jump",
  345. "配合上面的【优化图片浏览】更优雅浏览图片": "Collaborate with the optimization of image browsing above to browse images more elegantly",
  346. '需安装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>',
  347. 代码: "Code",
  348. 添加复制代码按钮: "Add Copy Code Button",
  349. 更优雅的复制: "More elegant replication",
  350. 快捷键: "Shortcut keys",
  351. "【F】键全屏、【Alt+Shift+F】键宽屏": "【F】 Key full screen, [Alt+Shift+F] key wide screen",
  352. 库: "Library",
  353. 脚本列表: "Script List",
  354. "请输入屏蔽规则,每行一个": "Please enter a blocking rule, one per line",
  355. 请求admin内容失败: "Request for admin content failed",
  356. 解析admin的源代码同步表单失败: "Failed to parse the source code of admin and synchronize the form",
  357. 源代码同步失败: "Source code synchronization failed",
  358. 获取用户信息失败: "Failed to obtain user information",
  359. 获取用户的收藏集失败: "Failed to retrieve user's collection",
  360. "解析Script Sets失败": "Parsing Script Sets failed",
  361. "获取收藏集{{setsId}}失败": "Failed to retrieve collection {{setsId}}",
  362. "获取表单元素#edit_script_set失败": "Failed to retrieve form element #edit_script_set",
  363. 更新收藏集表单请求失败: "Update collection form request failed",
  364. 请先在菜单中录入账号: "Please enter your account in the menu first",
  365. 请先在菜单中录入密码: "Please enter your password in the menu first",
  366. "获取csrf-token失败": "Failed to obtain csrf token",
  367. "正在登录中...": "Logging in...",
  368. "登录失败,请在控制台查看原因": "Login failed, please check the reason in the console",
  369. "登录成功,1s后自动跳转": "Login successful, automatically redirect after 1 second",
  370. "登录失败,可能是账号/密码错误,请在控制台查看原因": "Login failed, possibly due to incorrect account/password. Please check the reason in the console",
  371. "美化 历史版本 页面": "Beautify the historical version page",
  372. 未找到history_versions元素列表: "History_versions element list not found",
  373. "yyyy年MM月dd日 HH:mm:ss": "yyyy-MM-dd HH:mm:ss",
  374. "美化 Greasyfork Beautify脚本": "Beautify Greasyfork Beauty Script",
  375. "❌ 最多同时长传5张图": "❌ Upload up to 5 images simultaneously",
  376. "❌ 图片:{{name}} 大小:{{size}}": "❌ Image:{{name}} Size:{{size}}",
  377. "已过滤:{{oldCount}}": "Filtered:{{oldCount}}",
  378. 寻找引用: "Find references",
  379. 获取脚本id失败: "Failed to obtain script ID",
  380. 收藏: "Collection",
  381. 请先登录账号: "Please log in to your account first",
  382. 获取用户id失败: "Failed to obtain user ID",
  383. "获取收藏夹中...": "Get in favorites...",
  384. 收藏集: "Collection",
  385. "添加中...": "Adding...",
  386. "添加失败,{{selector}}元素不存在": "Add failed, {{selector}} element does not exist",
  387. "未找到{{selector}}元素": "{{selector}} element not found",
  388. 添加失败: "Add failed",
  389. 添加成功: "Successfully added",
  390. "删除中...": "Deleting...",
  391. 删除成功: "Delete successful",
  392. 添加: "Add in deletion",
  393. 刪除: "Delete",
  394. "拦截跳转:": "Intercept jump:",
  395. 今日检查: "Today's inspection",
  396. 复制代码: "Copy Code",
  397. "加载文件中...": "Loading files...",
  398. 复制成功: "Copy successful",
  399. "✅ 复制成功!": "✅ Copy successful!",
  400. "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}": "Current language: {{currentLocaleLanguage}}, switch to {{localeLanguage}} in 3 seconds",
  401. "导航至:": "Navigation to:",
  402. "请先登录账号!": "Please log in to your account first!",
  403. "获取信息中,请稍后...": "Obtaining information, please wait...",
  404. "获取成功,共 {{count}} 个": "Successfully obtained, a total of {{count}}",
  405. "评分:": "Rating:",
  406. "语言:": "Language:",
  407. "版本:": "Version:",
  408. "更新:": "Update:",
  409. 同步代码: "Synchronize Code",
  410. "同步中...": "Synchronizing...",
  411. 手动: "Manual",
  412. 自动: "Automatic",
  413. "同步方式:{{syncMode}}": "Synchronization method: {{syncMode}}",
  414. 同步成功: "Sync successful",
  415. 同步失败: "Sync failed",
  416. 该脚本未设置同步信息: "The script has not set synchronization information",
  417. "上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载": "Last reload time {{time}}, rejected repeated reloads within {{timeout}} seconds",
  418. "名称:": "Name:",
  419. "进度:": "Progress:",
  420. "未获取到【脚本列表】": "Unable to obtain [Script List]",
  421. "源代码同步成功,3秒后更新下一个": "Source code synchronization successful, update next one in 3 seconds",
  422. 全部更新失败: "All updates failed",
  423. "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}": "All updates completed<br>Success: {{successNums}}<br>Failure: {{failed Nums}}<br>Total: {{scriptUrlListLength}}",
  424. "⚙ 设置": "⚙ Setting",
  425. "{{SCRIPT_NAME}}-设置": "{{SCRIPT_NAME}}-Setting",
  426. 美化页面元素: "Beautify page elements",
  427. 美化历史版本页面: "Beautify the historical version page",
  428. "美化Greasyfork Beautify脚本": "Beautify Greasyfork Beauty Script",
  429. 获取表单csrfToken失败: "Failed to obtain form csrfToken",
  430. Toast配置: "Toast Config",
  431. Toast位置: "Toast position",
  432. 左上角: "Top left",
  433. 顶部: "Top",
  434. 右上角: "Top right",
  435. 左边: "Left",
  436. 中间: "Center",
  437. 右边: "Right",
  438. 左下角: "Bottom left",
  439. 底部: "Bottom",
  440. 右下角: "Bottom right",
  441. Toast显示在页面九宫格的位置: "Toast is displayed in the nine grid position on the page",
  442. 最多显示的数量: "Maximum number of displays",
  443. 限制Toast显示的数量: "Limit the number of Toast displays",
  444. 逆序弹出: "Reverse pop-up",
  445. 修改Toast弹出的顺序: "Modify the order in which Toast pops up",
  446. 该脚本已经在该收藏集中: "The script is already in this collection",
  447. 其它错误: "Ohter Error",
  448. 启用: "Enable",
  449. 开启后下面的过滤功能才会生效: "The following filtering features will only take effect after it is enabled",
  450. 屏蔽脚本: "Block script",
  451. 点击查看规则: "Click to view rules",
  452. 过滤: "Filter",
  453. 代码同步: "Code synchronization",
  454. 美化: "Beautify",
  455. 修复代码行号显示: "Fix code line number display",
  456. 修复代码行数超过1k行号显示不全问题: "Fix the problem that the code line number display is not complete when the number of lines exceeds 1k",
  457. "添加【寻找引用】按钮": "Add the button to find references",
  458. "在脚本栏添加按钮,一般用于搜索引用该库的相关脚本": "Add a button to the script bar, generally used to search for scripts that reference this library",
  459. "添加【收藏】按钮": "Add the button to collect",
  460. "在脚本栏添加按钮,一般用于快捷收藏该脚本/库": "Add a button to the script bar, generally used to quickly collect this script / library",
  461. 修复图片宽度显示问题: " Fix the problem that the picture width display is not complete",
  462. 修复图片在移动端宽度超出浏览器宽度问题: "Fix the problem that the picture width exceeds the browser width on mobile",
  463. "添加【今日检查】信息块": "Add the block of information of today's inspection",
  464. "在脚本信息栏添加【今日检查】信息块": "Add the block of information of today's inspection to the script information bar",
  465. "给Markdown添加【复制】按钮": "Add the button to copy to Markdown",
  466. "在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",
  467. 开启后下面的功能才会生效: "The following features will only take effect after it is enabled",
  468. 检测页面加载: "Detect page loading",
  469. "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面": "Detect whether the Greasyfork page is loaded normally. If the loading fails, the page will be automatically refreshed",
  470. 检测间隔: "Detection interval",
  471. "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面": "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",
  472. 美化顶部导航栏: "Beautify the top navigation bar",
  473. "可能会跟Greasyfork Beautify脚本有冲突": "Possible conflict with Greasymfork Beautify script",
  474. 美化脚本列表: "Beautify Script List",
  475. "双列显示且添加脚本卡片操作项(安装、收藏)": "Double column display and add script card operation items (installation, bookmarking)",
  476. 操作面板: "Operation Panel",
  477. "添加【操作面板】按钮": "Add [Operation Panel] button",
  478. "在脚本列表页面时为顶部导航栏添加【操作面板】按钮": "Add an 'Operation Panel' button to the top navigation bar on the script list page",
  479. 操作: "Operation",
  480. 安装此脚本: "Install this script",
  481. 脚本: "Scripts",
  482. 历史版本: "Historical version",
  483. 自定义已读颜色: "Customize read colors",
  484. 在讨论内生效: "Effective within the discussion",
  485. 用户: "Users",
  486. 控制台: "Console",
  487. "迁移【控制台】到顶部导航栏": "Migration of Console to Top Navigation Bar",
  488. "将【控制台】按钮移动到顶部导航栏,节省空间": "Move the 'Console' button to the top navigation bar to save space",
  489. 添加额外的标签按钮: "Add additional label button",
  490. "在版本下面添加【安装】、【查看代码】按钮": "Add 【 Install 】 and 【 View Code 】 buttons under the version",
  491. 查看代码: "View Code",
  492. "添加【过滤】按钮": "Add [Filter] button",
  493. "添加【举报】按钮": "Add [Report] button",
  494. "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效": "Add a 'Filter' button at the end of each discussion line. The filtering features needs to be enabled for it to take effect",
  495. "在每一行讨论的最后面添加【举报】按钮": "Add a Report button at the end of each line of discussion",
  496. 选择需要过滤的选项: "Select the options that need to be filtered",
  497. "确定{{type}}:{{filterId}}?": "Are you sure {{type}}:{{filterId}}?",
  498. "已删除:{{scriptId}}": "Deleted: {{scriptId}}",
  499. 帮助文档: "Help document",
  500. "请输入规则,每行一个": "Please enter a rule, one per line",
  501. 选择过滤的选项: "Select filtering options",
  502. "脚本id:{{text}}": "Script Id: {{text}}",
  503. "脚本名:{{text}}": "Script Name: {{text}}",
  504. "作者id:{{text}}": "Author Id: {{text}}",
  505. "作者名:{{text}}": "Author Name: {{text}}",
  506. "作用域:脚本、脚本搜索、用户主页": "Scope: Script, Script Search, User Homepage",
  507. "更新到 {{version}} 版本": "Update To {{version}} Version",
  508. "降级到 {{version}} 版本": "Downgrade to {{version}} Version",
  509. "重新安装 {{version}} 版本": "Reinstall {{version}} Version",
  510. "发布的用户id:{{text}}": "Published user ID: {{text}}",
  511. 自定义快捷键: "Customize shortcut keys",
  512. 点击录入快捷键: "Click on the input shortcut key",
  513. 快捷键发表回复: "Shortcut key to post reply",
  514. "在输入框内按下快捷发表回复,例如:{{key}}": "Press the shortcut to post a reply in the input box, for example: {{key}}",
  515. 请先执行当前的录入操作: "Please perform the current input operation first",
  516. 清空快捷键: "Clear shortcut keys",
  517. "请按下快捷键...": "Please press the shortcut key...",
  518. 成功录入: "Successful entry",
  519. "快捷键 {{key}} 已被 {{isUsedKey}} 占用": "The shortcut key {{key}} is already used by {{isUsedKey}}",
  520. 私聊: "Private Chat",
  521. 美化私信页面: "Beautify the private message page",
  522. 美化为左右对话模式: "Beautify as a left-right dialogue mode",
  523. "最后回复:": "Final response:",
  524. 进入: "Enter",
  525. 记住回复内容: "Remember the reply content",
  526. "监听表单内的textarea内容改变并存储到indexDB中,提交表单将清除保存的数据,误刷新页面时可动态恢复": "Monitor changes to the textarea content in the form and store it in the index database. Submitting the form will clear the saved data, and dynamic recovery can be achieved when the page is accidentally refreshed",
  527. 表单: "Forms",
  528. 自动清理空间: "Automatically clear space",
  529. 不清理: "Not cleaning",
  530. "{{value}} 天": "{{value}} day",
  531. "{{value}} 周": "{{value}} weeks",
  532. "{{value}} 个月": "{{value}} months",
  533. 半年: "half a year",
  534. 计算中: "In the process of calculation",
  535. 根据设置的间隔时间自动清理保存的回复内容: "Automatically clean up saved reply content according to the set interval time",
  536. "数据占用空间:{{size}}": "Data occupancy space: {{size}}",
  537. 当前存储的数据所占用的空间大小: "The size of the space occupied by the currently stored data",
  538. 清空: "Clear",
  539. 清理成功: "Cleanup successful",
  540. 清理失败: "Cleaning failed",
  541. "Url To WebhookUrl": "Url To WebhookUrl",
  542. 关闭: "Clsoe",
  543. "例如:": "Example: ",
  544. "结果:": "Result: ",
  545. 转换前: "Before Parse",
  546. 转换后: "Parse Result",
  547. 使用namespace查询脚本信息: "Use a namespace to query script information",
  548. 脚本管理: "Script management",
  549. "开启后检测已安装的脚本信息更准确,但是速度会更慢": "Detecting the installed script information is more accurate, but slower",
  550. 美化私信列表: "Beautify the private message list",
  551. 搜索: "Search",
  552. "新增【{{buttonText}}】按钮": "Added [{{buttonText}}] button",
  553. "该Checkbox按钮开启后,自动过滤出包含搜索关键词的脚本": "When the Checkbox button is turned on, it automatically filters out scripts that contain search terms",
  554. "名称-全词匹配": "Name - Full word match",
  555. "描述-全词匹配": "Description - Full word match",
  556. "作者名称-全词匹配": "Author name - Full word match",
  557. 获取举报表单信息失败: "Failed to obtain report form information. Procedure",
  558. 发送举报表单失败: "Failed to send the report form. Procedure",
  559. 举报: "Report",
  560. "举报讨论:": "Report discussion:",
  561. "举报脚本:": "Report script:",
  562. "举报用户:": "Report user:",
  563. "添加失败,表单数据中不包含该脚本": "Failed to add, script id not included in form data",
  564. "删除失败,表单数据中仍包含该脚本": "The deletion failed and the script is still included in the form data",
  565. "删除失败,{{selector}}元素不存在": "Failed to delete. {{selector}} element does not exist",
  566. "对比选中版本差异(monacoEditor)": "Compare the differences between selected versions (monacoEditor)",
  567. "正在加载monaco中...": "Loading monaco...",
  568. "正在获取对比文本中...": "Retrieving comparison text...",
  569. 代码对比: "Code Comparison",
  570. 添加代码对比按钮: "Add code comparison button",
  571. "版本号相同,不需要比较源码": "The version numbers are the same, no need to compare source code",
  572. 使用Monaco编辑器: "Use Monaco Editor"
  573. };
  574. const KEY = "GM_Panel";
  575. const ATTRIBUTE_INIT = "data-init";
  576. const ATTRIBUTE_KEY = "data-key";
  577. const ATTRIBUTE_DEFAULT_VALUE = "data-default-value";
  578. const ATTRIBUTE_INIT_MORE_VALUE = "data-init-more-value";
  579. const PROPS_STORAGE_API = "data-storage-api";
  580. const LanguageInit = function() {
  581. let settingPanel = _GM_getValue(KEY, {});
  582. let lng = settingPanel["setting-language"] || "zh-CN";
  583. i18next.init({
  584. lng,
  585. // lng: "zh-CN",
  586. fallbackLng: "zh-CN",
  587. resources: {
  588. "zh-CN": {
  589. translation: { ...zh_CN_language }
  590. },
  591. "en-US": {
  592. translation: { ...en_US_language }
  593. }
  594. }
  595. });
  596. };
  597. const CommonUtil = {
  598. /**
  599. * 添加屏蔽CSS
  600. * @param args
  601. * @example
  602. * addBlockCSS("")
  603. * addBlockCSS("","")
  604. * addBlockCSS(["",""])
  605. */
  606. addBlockCSS(...args) {
  607. let selectorList = [];
  608. if (args.length === 0) {
  609. return;
  610. }
  611. if (args.length === 1 && typeof args[0] === "string" && args[0].trim() === "") {
  612. return;
  613. }
  614. args.forEach((selector) => {
  615. if (Array.isArray(selector)) {
  616. selectorList = selectorList.concat(selector);
  617. } else {
  618. selectorList.push(selector);
  619. }
  620. });
  621. return addStyle(`${selectorList.join(",\n")}{display: none !important;}`);
  622. },
  623. /**
  624. * 设置GM_getResourceText的style内容
  625. * @param resourceMapData 资源数据
  626. * @example
  627. * setGMResourceCSS({
  628. * keyName: "ViewerCSS",
  629. * url: "https://example.com/example.css",
  630. * })
  631. */
  632. setGMResourceCSS(resourceMapData) {
  633. let cssText = typeof _GM_getResourceText === "function" ? _GM_getResourceText(resourceMapData.keyName) : "";
  634. if (typeof cssText === "string" && cssText) {
  635. addStyle(cssText);
  636. } else {
  637. CommonUtil.loadStyleLink(resourceMapData.url);
  638. }
  639. },
  640. /**
  641. * 添加<link>标签
  642. * @param url
  643. * @example
  644. * loadStyleLink("https://example.com/example.css")
  645. */
  646. async loadStyleLink(url) {
  647. let $link = document.createElement("link");
  648. $link.rel = "stylesheet";
  649. $link.type = "text/css";
  650. $link.href = url;
  651. domUtils.ready(() => {
  652. document.head.appendChild($link);
  653. });
  654. },
  655. /**
  656. * 添加<script>标签
  657. * @param url
  658. * @example
  659. * loadStyleLink("https://example.com/example.js")
  660. */
  661. async loadScript(url) {
  662. let $script = document.createElement("script");
  663. $script.src = url;
  664. return new Promise((resolve) => {
  665. $script.onload = () => {
  666. resolve(null);
  667. };
  668. (document.head || document.documentElement).appendChild($script);
  669. });
  670. },
  671. /**
  672. * 将url修复,例如只有search的链接修复为完整的链接
  673. *
  674. * 注意:不包括http转https
  675. * @param url 需要修复的链接
  676. * @example
  677. * 修复前:`/xxx/xxx?ss=ssss`
  678. * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss`
  679. * @example
  680. * 修复前:`//xxx/xxx?ss=ssss`
  681. * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss`
  682. * @example
  683. * 修复前:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss`
  684. * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss`
  685. * @example
  686. * 修复前:`xxx/xxx?ss=ssss`
  687. * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss`
  688. */
  689. fixUrl(url) {
  690. url = url.trim();
  691. if (url.match(/^http(s|):\/\//i)) {
  692. return url;
  693. } else {
  694. if (!url.startsWith("/")) {
  695. url += "/";
  696. }
  697. url = window.location.origin + url;
  698. return url;
  699. }
  700. },
  701. /**
  702. * http转https
  703. * @param url 需要修复的链接
  704. * @example
  705. * 修复前:
  706. * 修复后:
  707. * @example
  708. * 修复前:
  709. * 修复后:
  710. */
  711. fixHttps(url) {
  712. if (url.startsWith("https://")) {
  713. return url;
  714. }
  715. if (!url.startsWith("http://")) {
  716. return url;
  717. }
  718. let urlObj = new URL(url);
  719. urlObj.protocol = "https:";
  720. return urlObj.toString();
  721. },
  722. /**
  723. * 禁止页面滚动,默认锁定html和body
  724. * @example
  725. * lockScroll();
  726. * @example
  727. * lockScroll(document.body);
  728. */
  729. lockScroll(...args) {
  730. let $hidden = document.createElement("style");
  731. $hidden.innerHTML = /*css*/
  732. `
  733. .pops-overflow-hidden-important {
  734. overflow: hidden !important;
  735. }
  736. `;
  737. let $elList = [document.documentElement, document.body].concat(
  738. ...args || []
  739. );
  740. $elList.forEach(($el) => {
  741. $el.classList.add("pops-overflow-hidden-important");
  742. });
  743. (document.head || document.documentElement).appendChild($hidden);
  744. return {
  745. /**
  746. * 解除锁定
  747. */
  748. recovery() {
  749. $elList.forEach(($el) => {
  750. $el.classList.remove("pops-overflow-hidden-important");
  751. });
  752. $hidden.remove();
  753. }
  754. };
  755. }
  756. };
  757. LanguageInit();
  758. _GM_getValue(KEY, {});
  759. const _SCRIPT_NAME_ = i18next.t("GreasyFork优化");
  760. const utils = Utils.noConflict();
  761. const domUtils = DOMUtils.noConflict();
  762. const __pops = pops;
  763. const log = new utils.Log(
  764. _GM_info,
  765. _unsafeWindow.console || _monkeyWindow.console
  766. );
  767. const SCRIPT_NAME = ((_a = _GM_info == null ? void 0 : _GM_info.script) == null ? void 0 : _a.name) || _SCRIPT_NAME_;
  768. const DEBUG = false;
  769. log.config({
  770. debug: DEBUG,
  771. logMaxCount: 1e3,
  772. autoClearConsole: true,
  773. tag: true
  774. });
  775. Qmsg.config(
  776. Object.defineProperties(
  777. {
  778. html: true,
  779. autoClose: true,
  780. showClose: false
  781. },
  782. {
  783. position: {
  784. get() {
  785. return PopsPanel.getValue("qmsg-config-position", "bottom");
  786. }
  787. },
  788. maxNums: {
  789. get() {
  790. return PopsPanel.getValue("qmsg-config-maxnums", 5);
  791. }
  792. },
  793. showReverse: {
  794. get() {
  795. return PopsPanel.getValue("qmsg-config-showreverse", true);
  796. }
  797. },
  798. zIndex: {
  799. get() {
  800. let maxZIndex = Utils.getMaxZIndex();
  801. let popsMaxZIndex = pops.config.InstanceUtils.getPopsMaxZIndex().zIndex;
  802. return Utils.getMaxValue(maxZIndex, popsMaxZIndex) + 100;
  803. }
  804. }
  805. }
  806. )
  807. );
  808. const GM_Menu = new utils.GM_Menu({
  809. GM_getValue: _GM_getValue,
  810. GM_setValue: _GM_setValue,
  811. GM_registerMenuCommand: _GM_registerMenuCommand,
  812. GM_unregisterMenuCommand: _GM_unregisterMenuCommand
  813. });
  814. const httpx = new utils.Httpx(_GM_xmlhttpRequest);
  815. httpx.interceptors.response.use(void 0, (data) => {
  816. log.error("拦截器-请求错误", data);
  817. if (data.type === "onabort") {
  818. Qmsg.warning(i18next.t("请求取消"));
  819. } else if (data.type === "onerror") {
  820. Qmsg.error(i18next.t("请求异常"));
  821. } else if (data.type === "ontimeout") {
  822. Qmsg.error(i18next.t("请求超时"));
  823. } else {
  824. Qmsg.error(i18next.t("其它错误"));
  825. }
  826. return data;
  827. });
  828. httpx.config({
  829. logDetails: DEBUG
  830. });
  831. ({
  832. Object: {
  833. defineProperty: _unsafeWindow.Object.defineProperty
  834. },
  835. Function: {
  836. apply: _unsafeWindow.Function.prototype.apply,
  837. call: _unsafeWindow.Function.prototype.call
  838. },
  839. Element: {
  840. appendChild: _unsafeWindow.Element.prototype.appendChild
  841. },
  842. setTimeout: _unsafeWindow.setTimeout
  843. });
  844. const addStyle = utils.addStyle.bind(utils);
  845. const $ = document.querySelector.bind(document);
  846. const $$ = document.querySelectorAll.bind(document);
  847. const UIButton = function(text, description, buttonText, buttonIcon, buttonIsRightIcon, buttonIconIsLoading, buttonType, clickCallBack, afterAddToUListCallBack) {
  848. let result = {
  849. text,
  850. type: "button",
  851. description,
  852. buttonIcon,
  853. buttonIsRightIcon,
  854. buttonIconIsLoading,
  855. buttonType,
  856. buttonText,
  857. callback(event) {
  858. if (typeof clickCallBack === "function") {
  859. clickCallBack(event);
  860. }
  861. },
  862. afterAddToUListCallBack
  863. };
  864. return result;
  865. };
  866. const UIInput = function(text, key, defaultValue, description, changeCallBack, placeholder = "", isNumber, isPassword) {
  867. let result = {
  868. text,
  869. type: "input",
  870. isNumber: Boolean(isNumber),
  871. isPassword: Boolean(isPassword),
  872. props: {},
  873. attributes: {},
  874. description,
  875. getValue() {
  876. return this.props[PROPS_STORAGE_API].get(key, defaultValue);
  877. },
  878. callback(event, value) {
  879. this.props[PROPS_STORAGE_API].set(key, value);
  880. },
  881. placeholder
  882. };
  883. Reflect.set(result.attributes, ATTRIBUTE_KEY, key);
  884. Reflect.set(result.attributes, ATTRIBUTE_DEFAULT_VALUE, defaultValue);
  885. Reflect.set(result.props, PROPS_STORAGE_API, {
  886. get(key2, defaultValue2) {
  887. return PopsPanel.getValue(key2, defaultValue2);
  888. },
  889. set(key2, value) {
  890. PopsPanel.setValue(key2, value);
  891. }
  892. });
  893. return result;
  894. };
  895. const UISwitch = function(text, key, defaultValue, clickCallBack, description, afterAddToUListCallBack) {
  896. let result = {
  897. text,
  898. type: "switch",
  899. description,
  900. attributes: {},
  901. props: {},
  902. getValue() {
  903. return Boolean(
  904. this.props[PROPS_STORAGE_API].get(key, defaultValue)
  905. );
  906. },
  907. callback(event, __value) {
  908. let value = Boolean(__value);
  909. log.success(`${value ? "开启" : "关闭"} ${text}`);
  910. this.props[PROPS_STORAGE_API].set(key, value);
  911. },
  912. afterAddToUListCallBack
  913. };
  914. Reflect.set(result.attributes, ATTRIBUTE_KEY, key);
  915. Reflect.set(result.attributes, ATTRIBUTE_DEFAULT_VALUE, defaultValue);
  916. Reflect.set(result.props, PROPS_STORAGE_API, {
  917. get(key2, defaultValue2) {
  918. return PopsPanel.getValue(key2, defaultValue2);
  919. },
  920. set(key2, value) {
  921. PopsPanel.setValue(key2, value);
  922. }
  923. });
  924. return result;
  925. };
  926. const GreasyforkApi = {
  927. /**
  928. * 获取脚本统计数据
  929. * @param scriptId
  930. */
  931. async getScriptStats(scriptId) {
  932. let response = await httpx.get(`/scripts/${scriptId}/stats.json`, {
  933. fetch: true,
  934. allowInterceptConfig: false
  935. });
  936. log.info(response);
  937. if (!response.status) {
  938. log.error(i18next.t("获取脚本统计数据失败"));
  939. return;
  940. }
  941. let scriptStatsJSON = utils.toJSON(response.data.responseText);
  942. return scriptStatsJSON;
  943. },
  944. /**
  945. * 解析并获取admin内的源代码同步的配置表单
  946. * @param scriptId
  947. */
  948. async getSourceCodeSyncFormData(scriptId) {
  949. let response = await httpx.get(`/scripts/${scriptId}/admin`, {
  950. fetch: true,
  951. allowInterceptConfig: false
  952. });
  953. log.info(response);
  954. if (!response.status) {
  955. Qmsg.error(i18next.t("请求admin内容失败"));
  956. return;
  957. }
  958. let adminHTML = response.data.responseText;
  959. let adminHTMLElement = domUtils.parseHTML(adminHTML, false, true);
  960. let formElement = adminHTMLElement.querySelector("form.edit_script");
  961. if (!formElement) {
  962. Qmsg.error(i18next.t("解析admin的源代码同步表单失败"));
  963. return;
  964. }
  965. let formData = new FormData(formElement);
  966. return formData;
  967. },
  968. /**
  969. * 进行源代码同步,要求先getSourceCodeSyncFormData
  970. * @param scriptId
  971. * @param data
  972. */
  973. async sourceCodeSync(scriptId, data) {
  974. let response = await httpx.post(`/scripts/${scriptId}/sync_update`, {
  975. fetch: true,
  976. data,
  977. allowInterceptConfig: false
  978. });
  979. log.info(response);
  980. if (!response.status) {
  981. Qmsg.error(i18next.t("源代码同步失败"));
  982. return;
  983. }
  984. return response;
  985. },
  986. /**
  987. * 获取用户的信息,包括脚本列表、未上架的脚本、库
  988. */
  989. async getUserInfo(userId) {
  990. let response = await httpx.get(`/users/${userId}.json`, {
  991. fetch: true,
  992. allowInterceptConfig: false
  993. });
  994. log.success(response);
  995. if (!response.status) {
  996. Qmsg.error(i18next.t("获取用户信息失败"));
  997. return;
  998. }
  999. let data = utils.toJSON(response.data.responseText);
  1000. data["scriptList"] = [];
  1001. data["scriptLibraryList"] = [];
  1002. data["scripts"].forEach((scriptInfo) => {
  1003. if (scriptInfo["code_url"].endsWith(".user.js")) {
  1004. data["scriptList"].push(scriptInfo);
  1005. } else {
  1006. data["scriptLibraryList"].push(scriptInfo);
  1007. }
  1008. });
  1009. return data;
  1010. },
  1011. /**
  1012. * 获取用户的收藏集
  1013. * @param userId
  1014. */
  1015. async getUserCollection(userId) {
  1016. let response = await httpx.get(`/users/${userId}`, {
  1017. fetch: true,
  1018. allowInterceptConfig: false
  1019. });
  1020. log.info("获取用户的收藏集", response);
  1021. if (!response.status) {
  1022. Qmsg.error(i18next.t("获取用户的收藏集失败"));
  1023. return;
  1024. }
  1025. let respText = response.data.responseText;
  1026. let respDocument = domUtils.parseHTML(respText, true, true);
  1027. let userScriptSets = respDocument.querySelector("#user-script-sets");
  1028. if (!userScriptSets) {
  1029. log.error("解析Script Sets失败");
  1030. return;
  1031. }
  1032. let scriptSetsIdList = [];
  1033. userScriptSets.querySelectorAll("li").forEach((liElement) => {
  1034. var _a2;
  1035. let $ele = liElement.querySelector("a:last-child");
  1036. if (!$ele) {
  1037. return;
  1038. }
  1039. let setsUrl = $ele.href;
  1040. if (setsUrl.includes("?fav=1")) {
  1041. return;
  1042. }
  1043. let setsName = liElement.querySelector("a").innerText;
  1044. let setsId = (_a2 = setsUrl.match(/\/sets\/([\d]+)\//)) == null ? void 0 : _a2[1];
  1045. scriptSetsIdList.push({
  1046. id: setsId,
  1047. name: setsName
  1048. });
  1049. });
  1050. return scriptSetsIdList;
  1051. },
  1052. /**
  1053. * 获取某个收藏集的信息
  1054. * @param userId 用户id
  1055. * @param setsId 收藏集id
  1056. */
  1057. async getUserCollectionInfo(userId, setsId) {
  1058. let response = await httpx.get(`/users/${userId}/sets/${setsId}/edit`, {
  1059. fetch: true,
  1060. allowInterceptConfig: false
  1061. });
  1062. log.info(response);
  1063. if (!response.status) {
  1064. Qmsg.error(i18next.t("获取收藏集{{setsId}}失败", { setsId }));
  1065. return;
  1066. }
  1067. let respText = response.data.responseText;
  1068. let respDocument = domUtils.parseHTML(respText, true, true);
  1069. let $edit_script_set_form = respDocument.querySelector(
  1070. 'form[id^="edit_script_set"]'
  1071. );
  1072. if (!$edit_script_set_form) {
  1073. Qmsg.error(i18next.t("获取表单元素#edit_script_set失败"));
  1074. return;
  1075. }
  1076. let formData = new FormData($edit_script_set_form);
  1077. let csrfToken = respDocument.querySelector(
  1078. 'meta[name="csrf-token"]'
  1079. );
  1080. if (!csrfToken) {
  1081. Qmsg.error(i18next.t("获取表单csrfToken失败"));
  1082. return;
  1083. }
  1084. if (csrfToken.hasAttribute("content")) {
  1085. let authenticity_token = csrfToken.getAttribute("content");
  1086. if (authenticity_token) {
  1087. formData.set("authenticity_token", authenticity_token);
  1088. }
  1089. }
  1090. return formData;
  1091. },
  1092. /**
  1093. * 更新用户的某个收藏集的表单信息
  1094. * @param userId 用户id
  1095. * @param setsId 收藏集id
  1096. * @param data
  1097. */
  1098. async updateUserSetsInfo(userId, setsId, data) {
  1099. let response = await httpx.post(`/users/${userId}/sets/${setsId}`, {
  1100. fetch: true,
  1101. allowInterceptConfig: false,
  1102. headers: {
  1103. 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",
  1104. "Accept-Language": "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6",
  1105. "Cache-Control": "no-cache",
  1106. "Content-Type": "application/x-www-form-urlencoded",
  1107. Pragma: "no-cache"
  1108. },
  1109. fetchInit: {
  1110. referrerPolicy: "strict-origin-when-cross-origin"
  1111. },
  1112. data
  1113. });
  1114. log.info(response);
  1115. if (!response.status) {
  1116. Qmsg.error(i18next.t("更新收藏集表单请求失败"));
  1117. return;
  1118. }
  1119. let respText = response.data.responseText;
  1120. let respDocument = domUtils.parseHTML(respText, true, true);
  1121. return respDocument;
  1122. },
  1123. /**
  1124. * 切换语言
  1125. * @param url
  1126. */
  1127. async switchLanguage(url) {
  1128. let response = await httpx.get(url, {
  1129. fetch: true,
  1130. headers: {
  1131. "Upgrade-Insecure-Requests": "1"
  1132. }
  1133. });
  1134. log.info(response);
  1135. if (!response.status) {
  1136. return;
  1137. }
  1138. }
  1139. };
  1140. const GreasyforkRouter = {
  1141. /**
  1142. * 主页
  1143. *
  1144. * + /zh-CN
  1145. */
  1146. isHome() {
  1147. return window.location.pathname.split("/").length <= 2 && window.location.search === "";
  1148. },
  1149. /**
  1150. * 代码页面
  1151. *
  1152. * + /code...
  1153. */
  1154. isCode() {
  1155. var _a2;
  1156. return Boolean((_a2 = window.location.pathname.split("/")) == null ? void 0 : _a2.includes("code"));
  1157. },
  1158. /**
  1159. * 代码页面
  1160. *
  1161. * (严格比较)
  1162. *
  1163. * + /code
  1164. * + /code/
  1165. */
  1166. isCodeStrict() {
  1167. return Boolean(window.location.pathname.match(/\/code(\/|)$/));
  1168. },
  1169. /**
  1170. * 版本页面
  1171. *
  1172. * (严格比较)
  1173. *
  1174. * + /version
  1175. * + /version/
  1176. */
  1177. isVersion() {
  1178. return Boolean(window.location.pathname.match(/\/versions(\/|)$/));
  1179. },
  1180. /**
  1181. * 用户
  1182. *
  1183. * + /users/...
  1184. */
  1185. isUsers() {
  1186. return Boolean(window.location.pathname.match(/\/.+\/users\/.+/gi));
  1187. },
  1188. /**
  1189. * 私聊用户页面,可能是全部私信页面,也可能是某个用户的私信页面
  1190. *
  1191. * + /conversations...
  1192. */
  1193. isUsersConversations() {
  1194. return this.isUsers() && Boolean(window.location.pathname.includes("/conversations"));
  1195. },
  1196. /**
  1197. * 私聊xxx用户页面
  1198. *
  1199. * + /conversations/111...
  1200. */
  1201. isUsersConversationsWithSomeUser() {
  1202. return this.isUsersConversations() && Boolean(window.location.pathname.match(/\/conversations\/[\d]+/));
  1203. },
  1204. /**
  1205. * 脚本页面(单个脚本的页面)
  1206. *
  1207. * + /scripts/111...
  1208. */
  1209. isScript() {
  1210. return Boolean(window.location.pathname.match(/\/scripts\/[\d+]/));
  1211. },
  1212. /**
  1213. * 脚本管理页面
  1214. *
  1215. * + /scripts/.../admin
  1216. */
  1217. isScriptAdmin() {
  1218. return Boolean(window.location.pathname.endsWith("/admin"));
  1219. },
  1220. /**
  1221. * 脚本列表页面
  1222. *
  1223. * (严格比较)
  1224. *
  1225. * + /scripts
  1226. * + /scripts/
  1227. */
  1228. isScriptList() {
  1229. return Boolean(window.location.pathname.match(/\/scripts(\/|)$/));
  1230. },
  1231. /**
  1232. * 脚本列表-按域名
  1233. *
  1234. * + /scripts/by-site...
  1235. */
  1236. isScriptsBySite() {
  1237. return Boolean(window.location.pathname.match("/scripts/by-site"));
  1238. },
  1239. /**
  1240. * 脚本反馈
  1241. *
  1242. * + /scripts/xxxx/feedback
  1243. */
  1244. isScriptsFeedback() {
  1245. return this.isScript() && window.location.pathname.match(/\/feedback(\/|)$/i);
  1246. },
  1247. /**
  1248. * 库列表页面
  1249. *
  1250. * (严格比较)
  1251. *
  1252. * + /libraries
  1253. * + /libraries/
  1254. */
  1255. isScriptLibraryList() {
  1256. return Boolean(window.location.pathname.match(/\/libraries(\/|)$/));
  1257. },
  1258. /**
  1259. * 脚本搜索结果页面
  1260. *
  1261. * + /scripts?q=
  1262. */
  1263. isScriptSearch() {
  1264. let searchParams = new URLSearchParams(window.location.search);
  1265. return this.isScriptList() && searchParams.has("q");
  1266. },
  1267. /**
  1268. * 脚本代码搜索页面
  1269. *
  1270. * (严格比较)
  1271. *
  1272. * + /code-search
  1273. * + /code-search/
  1274. */
  1275. isScriptCodeSearch() {
  1276. return Boolean(window.location.pathname.match(/\/code-search(\/|)$/));
  1277. },
  1278. /**
  1279. * 讨论页面
  1280. *
  1281. * (严格比较)
  1282. *
  1283. * + /discussions
  1284. * + /discussions/
  1285. */
  1286. isDiscuessions() {
  1287. return Boolean(window.location.pathname.match(/\/discussions(\/|)$/));
  1288. },
  1289. /**
  1290. * 图片资源页面
  1291. */
  1292. isImageSource() {
  1293. return window.location.pathname.startsWith("/vite/assets");
  1294. }
  1295. };
  1296. const GreasyforkUrlUtils = {
  1297. /**
  1298. * 获取脚本安装的链接
  1299. * @param scriptId
  1300. * @param scriptVersion
  1301. * @param scriptName
  1302. * @returns
  1303. */
  1304. getInstallUrl(scriptId, scriptVersion, scriptName) {
  1305. if (utils.isNotNull(scriptName)) {
  1306. scriptName = "/" + scriptName;
  1307. } else {
  1308. scriptName = "";
  1309. }
  1310. return `https://update.greasyfork.org/scripts/${scriptId}/${scriptVersion}${scriptName}.user.js`;
  1311. },
  1312. /**
  1313. * 获取脚本的代码页面链接
  1314. * @param scriptId
  1315. * @param scriptVersion
  1316. * @returns
  1317. */
  1318. getCodeUrl(scriptId, scriptVersion) {
  1319. if (utils.isNull(scriptVersion)) {
  1320. scriptVersion = "";
  1321. }
  1322. return `https://greasyfork.org/scripts/${scriptId}/code?version=${scriptVersion}`;
  1323. },
  1324. /**
  1325. * 获取代码搜索地址
  1326. * @param url
  1327. */
  1328. getCodeSearchUrl(url) {
  1329. return "https://greasyfork.org/zh-CN/scripts/code-search?c=" + url;
  1330. },
  1331. /**
  1332. * 获取脚本的信息
  1333. * @param scriptId 脚本id
  1334. */
  1335. getScriptInfoUrl(scriptId) {
  1336. return `https://greasyfork.org/scripts/${scriptId}.json`;
  1337. },
  1338. /**
  1339. * 获取管理地址
  1340. * @param url
  1341. */
  1342. getAdminUrl(url) {
  1343. return url + "/admin";
  1344. },
  1345. /**
  1346. * 从字符串中提取Id
  1347. * @param text
  1348. * @default window.location.pathname
  1349. */
  1350. getScriptId(text) {
  1351. var _a2, _b;
  1352. return (_b = (_a2 = text || window.location.pathname) == null ? void 0 : _a2.match(
  1353. /\/scripts\/([\d]+)/i
  1354. )) == null ? void 0 : _b[1];
  1355. },
  1356. /**
  1357. * 从字符串中提取用户id
  1358. * @param text
  1359. * @default window.location.pathname
  1360. */
  1361. getUserId(text) {
  1362. var _a2;
  1363. return (_a2 = (text || window.location.pathname).match(/\/users\/([\d]+)/i)) == null ? void 0 : _a2[1];
  1364. },
  1365. /**
  1366. * 获取举报地址
  1367. */
  1368. getReportUrl(item_class, item_id) {
  1369. return `${window.location.origin}/reports/new?item_class=${item_class}&item_id=${item_id}`;
  1370. },
  1371. /**
  1372. * 从字符串中提取脚本名
  1373. * @param text
  1374. */
  1375. getScriptName(text) {
  1376. let pathname = window.location.pathname;
  1377. if (text != null) {
  1378. pathname = new URL(text).pathname;
  1379. }
  1380. pathname = decodeURIComponent(pathname);
  1381. let pathnameSplit = pathname.split("/");
  1382. for (const name of pathnameSplit) {
  1383. let nameMatch = name.match(/[\d]+/);
  1384. if (nameMatch && nameMatch.length) {
  1385. return nameMatch[1];
  1386. }
  1387. }
  1388. },
  1389. /**
  1390. * 获取需要切换语言的Url
  1391. * @default "zh-CN"
  1392. */
  1393. getSwitchLanguageUrl(localeLanguage = "zh-CN") {
  1394. let url = window.location.origin;
  1395. let urlSplit = window.location.pathname.split("/");
  1396. urlSplit[1] = localeLanguage;
  1397. url = url + urlSplit.join("/");
  1398. url += window.location.search;
  1399. if (window.location.search === "") {
  1400. url += "?locale_override=1";
  1401. } else if (!window.location.search.includes("locale_override=1")) {
  1402. url += "&locale_override=1";
  1403. }
  1404. return url;
  1405. }
  1406. };
  1407. const GreasyforkMenu = {
  1408. /**
  1409. * @class
  1410. */
  1411. menu: GM_Menu,
  1412. /**
  1413. * 当前是否已登录
  1414. */
  1415. isLogin: false,
  1416. /**
  1417. * 初始化环境变量
  1418. */
  1419. initEnv() {
  1420. let userLinkElement = this.getUserLinkElement();
  1421. this.isLogin = Boolean(userLinkElement);
  1422. },
  1423. /**
  1424. * 获取当前登录用户的a标签元素
  1425. */
  1426. getUserLinkElement() {
  1427. return document.querySelector(
  1428. "#nav-user-info span.user-profile-link a"
  1429. );
  1430. },
  1431. /**
  1432. * 更新脚本
  1433. * @param scriptUrlList
  1434. */
  1435. async updateScript(scriptUrlList) {
  1436. let getLoadingHTML = function(scriptName, progress = 1) {
  1437. return `
  1438. <div style="display: flex;flex-direction: column;align-items: flex-start;">
  1439. <div style="height: 30px;line-height: 30px;">${i18next.t(
  1440. "名称:"
  1441. )}${scriptName}</div>
  1442. <div style="height: 30px;line-height: 30px;">${i18next.t(
  1443. "进度:"
  1444. )}${progress}/${scriptUrlList.length}</div>
  1445. </div>`;
  1446. };
  1447. if (utils.isNull(scriptUrlList)) {
  1448. Qmsg.error(i18next.t("未获取到【脚本列表】"));
  1449. } else {
  1450. let loading = Qmsg.loading(
  1451. getLoadingHTML(
  1452. GreasyforkUrlUtils.getScriptName(scriptUrlList[0])
  1453. ),
  1454. {
  1455. html: true
  1456. }
  1457. );
  1458. let successNums = 0;
  1459. let failedNums = 0;
  1460. for (let index = 0; index < scriptUrlList.length; index++) {
  1461. let scriptUrl = scriptUrlList[index];
  1462. let scriptId = GreasyforkUrlUtils.getScriptId(scriptUrl);
  1463. log.success("更新:" + scriptUrl);
  1464. let scriptName = GreasyforkUrlUtils.getScriptName(scriptUrl);
  1465. loading.setHTML(getLoadingHTML(scriptName, index + 1));
  1466. let codeSyncFormData = await GreasyforkApi.getSourceCodeSyncFormData(
  1467. scriptId
  1468. );
  1469. if (codeSyncFormData) {
  1470. let syncUpdateStatus = await GreasyforkApi.sourceCodeSync(
  1471. scriptId,
  1472. codeSyncFormData
  1473. );
  1474. if (syncUpdateStatus) {
  1475. Qmsg.success(i18next.t("源代码同步成功,3秒后更新下一个"));
  1476. await utils.sleep(3e3);
  1477. successNums++;
  1478. } else {
  1479. Qmsg.error(i18next.t("源代码同步失败"));
  1480. failedNums++;
  1481. }
  1482. } else {
  1483. Qmsg.error(i18next.t("源代码同步失败"));
  1484. failedNums++;
  1485. }
  1486. }
  1487. loading.close();
  1488. if (successNums === 0) {
  1489. Qmsg.error(i18next.t("全部更新失败"));
  1490. } else {
  1491. Qmsg.success(
  1492. i18next.t(
  1493. "全部更新完毕<br >成功:{{successNums}}<br >失败:{{failedNums}}<br >总计:{{scriptUrlListLength}}",
  1494. {
  1495. successNums,
  1496. failedNums,
  1497. scriptUrlListLength: scriptUrlList.length
  1498. }
  1499. ),
  1500. {
  1501. html: true
  1502. }
  1503. );
  1504. }
  1505. }
  1506. },
  1507. /**
  1508. * 处理本地的goto事件
  1509. */
  1510. handleLocalGotoCallBack() {
  1511. if (PopsPanel.getValue("goto_updateSettingsAndSynchronize_scriptList")) {
  1512. PopsPanel.deleteValue("goto_updateSettingsAndSynchronize_scriptList");
  1513. if (!GreasyforkRouter.isUsers()) {
  1514. PopsPanel.setValue(
  1515. "goto_updateSettingsAndSynchronize_scriptList",
  1516. true
  1517. );
  1518. if (GreasyforkMenu.getUserLinkElement()) {
  1519. Qmsg.success(i18next.t("前往用户主页"));
  1520. window.location.href = GreasyforkMenu.getUserLinkElement().href;
  1521. } else {
  1522. Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
  1523. }
  1524. return;
  1525. }
  1526. let scriptUrlList = [];
  1527. document.querySelectorAll(
  1528. "#user-script-list-section li a.script-link"
  1529. ).forEach((item) => {
  1530. scriptUrlList = scriptUrlList.concat(
  1531. GreasyforkUrlUtils.getAdminUrl(item.href)
  1532. );
  1533. });
  1534. GreasyforkMenu.updateScript(scriptUrlList);
  1535. } else if (PopsPanel.getValue("goto_updateSettingsAndSynchronize_unlistedScriptList")) {
  1536. PopsPanel.deleteValue(
  1537. "goto_updateSettingsAndSynchronize_unlistedScriptList"
  1538. );
  1539. if (!GreasyforkRouter.isUsers()) {
  1540. PopsPanel.setValue(
  1541. "goto_updateSettingsAndSynchronize_unlistedScriptList",
  1542. true
  1543. );
  1544. if (GreasyforkMenu.getUserLinkElement()) {
  1545. Qmsg.success(i18next.t("前往用户主页"));
  1546. window.location.href = GreasyforkMenu.getUserLinkElement().href;
  1547. } else {
  1548. Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
  1549. }
  1550. return;
  1551. }
  1552. let scriptUrlList = [];
  1553. document.querySelectorAll(
  1554. "#user-unlisted-script-list li a.script-link"
  1555. ).forEach((item) => {
  1556. scriptUrlList = scriptUrlList.concat(
  1557. GreasyforkUrlUtils.getAdminUrl(item.href)
  1558. );
  1559. });
  1560. GreasyforkMenu.updateScript(scriptUrlList);
  1561. } else if (PopsPanel.getValue("goto_updateSettingsAndSynchronize_libraryScriptList")) {
  1562. PopsPanel.deleteValue(
  1563. "goto_updateSettingsAndSynchronize_libraryScriptList"
  1564. );
  1565. if (!GreasyforkRouter.isUsers()) {
  1566. PopsPanel.setValue(
  1567. "goto_updateSettingsAndSynchronize_libraryScriptList",
  1568. true
  1569. );
  1570. if (GreasyforkMenu.getUserLinkElement()) {
  1571. Qmsg.success(i18next.t("前往用户主页"));
  1572. window.location.href = GreasyforkMenu.getUserLinkElement().href;
  1573. } else {
  1574. Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
  1575. }
  1576. return;
  1577. }
  1578. let scriptUrlList = [];
  1579. document.querySelectorAll(
  1580. "#user-library-script-list li a.script-link"
  1581. ).forEach((item) => {
  1582. scriptUrlList = scriptUrlList.concat(
  1583. GreasyforkUrlUtils.getAdminUrl(item.href)
  1584. );
  1585. });
  1586. GreasyforkMenu.updateScript(scriptUrlList);
  1587. }
  1588. }
  1589. };
  1590. const UISelect = function(text, key, defaultValue, data, callback, description) {
  1591. let selectData = [];
  1592. if (typeof data === "function") {
  1593. selectData = data();
  1594. } else {
  1595. selectData = data;
  1596. }
  1597. let result = {
  1598. text,
  1599. type: "select",
  1600. description,
  1601. attributes: {},
  1602. props: {},
  1603. getValue() {
  1604. return this.props[PROPS_STORAGE_API].get(key, defaultValue);
  1605. },
  1606. callback(event, isSelectedValue, isSelectedText) {
  1607. let value = isSelectedValue;
  1608. log.info(`选择:${isSelectedText}`);
  1609. this.props[PROPS_STORAGE_API].set(key, value);
  1610. if (typeof callback === "function") {
  1611. callback(event, value, isSelectedText);
  1612. }
  1613. },
  1614. data: selectData
  1615. };
  1616. Reflect.set(result.attributes, ATTRIBUTE_KEY, key);
  1617. Reflect.set(result.attributes, ATTRIBUTE_DEFAULT_VALUE, defaultValue);
  1618. Reflect.set(result.props, PROPS_STORAGE_API, {
  1619. get(key2, defaultValue2) {
  1620. return PopsPanel.getValue(key2, defaultValue2);
  1621. },
  1622. set(key2, value) {
  1623. PopsPanel.setValue(key2, value);
  1624. }
  1625. });
  1626. return result;
  1627. };
  1628. const UIButtonShortCut = function(text, description, key, defaultValue, defaultButtonText, buttonType = "default", shortCut) {
  1629. let __defaultButtonText = typeof defaultButtonText === "function" ? defaultButtonText() : defaultButtonText;
  1630. if (typeof defaultValue === "object") {
  1631. shortCut.initConfig(key, defaultValue);
  1632. }
  1633. let getButtonText = () => {
  1634. return shortCut.getShowText(key, __defaultButtonText);
  1635. };
  1636. let result = UIButton(
  1637. text,
  1638. description,
  1639. getButtonText,
  1640. "keyboard",
  1641. false,
  1642. false,
  1643. buttonType,
  1644. async (event) => {
  1645. var _a2;
  1646. let $click = event.target;
  1647. let $btn = (_a2 = $click.closest(".pops-panel-button")) == null ? void 0 : _a2.querySelector("span");
  1648. if (shortCut.isWaitPress) {
  1649. Qmsg.warning("请先执行当前的录入操作");
  1650. return;
  1651. }
  1652. if (shortCut.hasOptionValue(key)) {
  1653. shortCut.emptyOption(key);
  1654. Qmsg.success("清空快捷键");
  1655. } else {
  1656. let loadingQmsg = Qmsg.loading("请按下快捷键...", {
  1657. showClose: true
  1658. });
  1659. let {
  1660. status,
  1661. option,
  1662. key: isUsedKey
  1663. } = await shortCut.enterShortcutKeys(key);
  1664. loadingQmsg.close();
  1665. if (status) {
  1666. log.success("成功录入快捷键", option);
  1667. Qmsg.success("成功录入");
  1668. } else {
  1669. Qmsg.error(
  1670. `快捷键 ${shortCut.translateKeyboardValueToButtonText(
  1671. option
  1672. )} 已被 ${isUsedKey} 占用`
  1673. );
  1674. }
  1675. }
  1676. $btn.innerHTML = getButtonText();
  1677. }
  1678. );
  1679. result.attributes = {};
  1680. Reflect.set(result.attributes, ATTRIBUTE_INIT, () => {
  1681. return false;
  1682. });
  1683. return result;
  1684. };
  1685. class ShortCut {
  1686. constructor(key) {
  1687. /** 存储的键 */
  1688. __publicField(this, "key", "short-cut");
  1689. /** 是否存在等待按下的按键 */
  1690. __publicField(this, "isWaitPress", false);
  1691. if (typeof key === "string") {
  1692. this.key = key;
  1693. }
  1694. }
  1695. /**
  1696. * 初始化配置默认值
  1697. */
  1698. initConfig(key, option) {
  1699. if (this.hasOption(key)) ;
  1700. else {
  1701. this.setOption(key, option);
  1702. }
  1703. }
  1704. /** 获取存储的键 */
  1705. getStorageKey() {
  1706. return this.key;
  1707. }
  1708. /**
  1709. * 获取本地存储的所有值
  1710. */
  1711. getLocalAllOptions() {
  1712. return _GM_getValue(this.key, []);
  1713. }
  1714. /**
  1715. * 判断是否存在该配置
  1716. * @param key 键
  1717. */
  1718. hasOption(key) {
  1719. let localOptions = this.getLocalAllOptions();
  1720. let findOption = localOptions.find((item) => item.key === key);
  1721. return !!findOption;
  1722. }
  1723. /**
  1724. * 判断是否存在该配置的value值
  1725. * @param key 键
  1726. */
  1727. hasOptionValue(key) {
  1728. if (this.hasOption(key)) {
  1729. let option = this.getOption(key);
  1730. return !((option == null ? void 0 : option.value) == null);
  1731. } else {
  1732. return false;
  1733. }
  1734. }
  1735. /**
  1736. * 获取配置
  1737. * @param key 键
  1738. * @param defaultValue 默认值
  1739. */
  1740. getOption(key, defaultValue) {
  1741. let localOptions = this.getLocalAllOptions();
  1742. let findOption = localOptions.find((item) => item.key === key);
  1743. return findOption ?? defaultValue;
  1744. }
  1745. /**
  1746. * 设置配置
  1747. * @param key 键
  1748. * @param value 配置
  1749. */
  1750. setOption(key, value) {
  1751. let localOptions = this.getLocalAllOptions();
  1752. let findIndex = localOptions.findIndex((item) => item.key === key);
  1753. if (findIndex == -1) {
  1754. localOptions.push({
  1755. key,
  1756. value
  1757. });
  1758. } else {
  1759. Reflect.set(localOptions[findIndex], "value", value);
  1760. }
  1761. _GM_setValue(this.key, localOptions);
  1762. }
  1763. /**
  1764. * 清空当前已有配置录入的值
  1765. * @param key
  1766. */
  1767. emptyOption(key) {
  1768. let result = false;
  1769. let localOptions = this.getLocalAllOptions();
  1770. let findIndex = localOptions.findIndex((item) => item.key === key);
  1771. if (findIndex !== -1) {
  1772. localOptions[findIndex].value = null;
  1773. result = true;
  1774. }
  1775. _GM_setValue(this.key, localOptions);
  1776. return result;
  1777. }
  1778. /**
  1779. * 删除配置
  1780. * @param key 键
  1781. */
  1782. deleteOption(key) {
  1783. let result = false;
  1784. let localValue = this.getLocalAllOptions();
  1785. let findValueIndex = localValue.findIndex((item) => item.key === key);
  1786. if (findValueIndex !== -1) {
  1787. localValue.splice(findValueIndex, 1);
  1788. result = true;
  1789. }
  1790. _GM_setValue(this.key, localValue);
  1791. return result;
  1792. }
  1793. /**
  1794. * 把配置的快捷键转成文字
  1795. * @param keyboardValue
  1796. * @returns
  1797. */
  1798. translateKeyboardValueToButtonText(keyboardValue) {
  1799. let result = "";
  1800. keyboardValue.ohterCodeList.forEach((ohterCodeKey) => {
  1801. result += utils.stringTitleToUpperCase(ohterCodeKey, true) + " + ";
  1802. });
  1803. result += utils.stringTitleToUpperCase(keyboardValue.keyName);
  1804. return result;
  1805. }
  1806. /**
  1807. * 获取快捷键显示的文字
  1808. * @param key 本地存储的快捷键键名
  1809. * @param defaultShowText 默认显示的文字
  1810. */
  1811. getShowText(key, defaultShowText) {
  1812. if (this.hasOption(key)) {
  1813. let localOption = this.getOption(key);
  1814. if (localOption.value == null) {
  1815. return defaultShowText;
  1816. } else {
  1817. return this.translateKeyboardValueToButtonText(localOption.value);
  1818. }
  1819. } else {
  1820. return defaultShowText;
  1821. }
  1822. }
  1823. /**
  1824. * 录入快捷键
  1825. * @param key 本地存储的快捷键键名
  1826. */
  1827. async enterShortcutKeys(key) {
  1828. return new Promise((resolve) => {
  1829. this.isWaitPress = true;
  1830. let keyboardListener = domUtils.listenKeyboard(
  1831. window,
  1832. "keyup",
  1833. (keyName, keyValue, ohterCodeList) => {
  1834. const currentOption = {
  1835. keyName,
  1836. keyValue,
  1837. ohterCodeList
  1838. };
  1839. const shortcutJSONString = JSON.stringify(currentOption);
  1840. const allOptions = this.getLocalAllOptions();
  1841. for (let index = 0; index < allOptions.length; index++) {
  1842. let localValue = allOptions[index];
  1843. if (localValue.key === key) {
  1844. continue;
  1845. }
  1846. const localShortCutJSONString = JSON.stringify(localValue.value);
  1847. let isUsedByOtherOption = false;
  1848. if (localValue.value != null && shortcutJSONString === localShortCutJSONString) {
  1849. isUsedByOtherOption = true;
  1850. }
  1851. if (isUsedByOtherOption) {
  1852. this.isWaitPress = false;
  1853. keyboardListener.removeListen();
  1854. resolve({
  1855. status: false,
  1856. key: localValue.key,
  1857. option: currentOption
  1858. });
  1859. return;
  1860. }
  1861. }
  1862. this.setOption(key, currentOption);
  1863. this.isWaitPress = false;
  1864. keyboardListener.removeListen();
  1865. resolve({
  1866. status: true,
  1867. key,
  1868. option: currentOption
  1869. });
  1870. }
  1871. );
  1872. });
  1873. }
  1874. /**
  1875. * 初始化全局键盘监听
  1876. * @param shortCutOption 快捷键配置 一般是{ "键名": { callback: ()=>{}}},键名是本地存储的自定义快捷键的键名
  1877. */
  1878. initGlobalKeyboardListener(shortCutOption) {
  1879. let localOptions = this.getLocalAllOptions();
  1880. if (!localOptions.length) {
  1881. log.warn("没有设置快捷键");
  1882. return;
  1883. }
  1884. let that = this;
  1885. function setListenKeyboard($ele, option) {
  1886. domUtils.listenKeyboard(
  1887. $ele,
  1888. "keydown",
  1889. (keyName, keyValue, ohterCodeList) => {
  1890. if (that.isWaitPress) {
  1891. return;
  1892. }
  1893. localOptions = that.getLocalAllOptions();
  1894. let findShortcutIndex = localOptions.findIndex((item) => {
  1895. let option2 = item.value;
  1896. let tempOption = {
  1897. keyName,
  1898. keyValue,
  1899. ohterCodeList
  1900. };
  1901. if (JSON.stringify(option2) === JSON.stringify(tempOption)) {
  1902. return item;
  1903. }
  1904. });
  1905. if (findShortcutIndex != -1) {
  1906. let findShortcut = localOptions[findShortcutIndex];
  1907. log.info("调用快捷键", findShortcut);
  1908. if (findShortcut.key in option) {
  1909. option[findShortcut.key].callback();
  1910. }
  1911. }
  1912. }
  1913. );
  1914. }
  1915. let WindowShortCutOption = {};
  1916. let ElementShortCutOption = {};
  1917. Object.keys(shortCutOption).forEach((localKey) => {
  1918. let option = shortCutOption[localKey];
  1919. if (option.target == null || typeof option.target === "string" && option.target === "") {
  1920. option.target = "window";
  1921. }
  1922. if (option.target === "window") {
  1923. Reflect.set(WindowShortCutOption, localKey, option);
  1924. } else {
  1925. Reflect.set(ElementShortCutOption, localKey, option);
  1926. }
  1927. });
  1928. setListenKeyboard(window, WindowShortCutOption);
  1929. domUtils.ready(() => {
  1930. Object.keys(ElementShortCutOption).forEach(async (localKey) => {
  1931. let option = ElementShortCutOption[localKey];
  1932. if (typeof option.target === "string") {
  1933. utils.waitNode(option.target, 1e4).then(($ele) => {
  1934. if (!$ele) {
  1935. return;
  1936. }
  1937. let __option = {};
  1938. Reflect.set(__option, localKey, option);
  1939. setListenKeyboard($ele, __option);
  1940. });
  1941. } else if (typeof option.target === "function") {
  1942. let target = await option.target();
  1943. if (target == null) {
  1944. return;
  1945. }
  1946. let __option = {};
  1947. Reflect.set(__option, localKey, option);
  1948. setListenKeyboard(target, __option);
  1949. } else {
  1950. let __option = {};
  1951. Reflect.set(__option, localKey, option);
  1952. setListenKeyboard(option.target, __option);
  1953. }
  1954. });
  1955. });
  1956. }
  1957. }
  1958. const GreasyforkShortCut = {
  1959. shortCut: new ShortCut(),
  1960. shortOption: {
  1961. "gf-quickReply": {
  1962. target: () => {
  1963. let $commentText = document.querySelector("form textarea");
  1964. let $replyBtn = document.querySelector(
  1965. 'input[name="commit"][type="submit"]'
  1966. );
  1967. if (!$commentText) {
  1968. log.error("页面不存在输入框");
  1969. return;
  1970. } else if (!$replyBtn) {
  1971. log.error("页面不存在【发表回复】按钮");
  1972. return;
  1973. }
  1974. log.success("监听快捷键:gf-quickReply");
  1975. return $commentText;
  1976. },
  1977. callback() {
  1978. if (document.activeElement) {
  1979. let $parentForm = document.activeElement.closest("form");
  1980. if (!$parentForm) {
  1981. log.error("当前activeElement不在表单内,无法触发快捷键");
  1982. return;
  1983. }
  1984. let $replyBtnList = $parentForm.querySelectorAll(
  1985. 'input[name="commit"][type="submit"]'
  1986. );
  1987. if (!$replyBtnList.length) {
  1988. log.error("表单内不存在【发表回复】按钮");
  1989. return;
  1990. }
  1991. if ($replyBtnList.length > 1) {
  1992. log.warn("表单内存在多个【发表回复】按钮,只触发第一个");
  1993. }
  1994. $replyBtnList[0].click();
  1995. } else {
  1996. log.error("当前页面没有激活元素,无法触发快捷键");
  1997. }
  1998. }
  1999. }
  2000. },
  2001. init() {
  2002. this.shortCut.initGlobalKeyboardListener(this.shortOption);
  2003. }
  2004. };
  2005. const GreasyforkRememberFormTextArea = {
  2006. $key: {
  2007. DB_KEY: "data"
  2008. },
  2009. $data: {
  2010. db: null
  2011. },
  2012. init() {
  2013. this.$data.db = this.getDB();
  2014. PopsPanel.execMenuOnce("rememberReplyContent", () => {
  2015. this.rememberReplyContent();
  2016. });
  2017. PopsPanel.execMenu("gf-autoClearRememberReplayContent", (value) => {
  2018. this.autoClearRememberReplayContent(value);
  2019. });
  2020. },
  2021. /**
  2022. * 获取数据库连接对象
  2023. */
  2024. getDB() {
  2025. const dbName = "reply_record";
  2026. const storeName = "textarea_text";
  2027. const dbVersion = 2;
  2028. return new utils.indexedDB(dbName, storeName, dbVersion);
  2029. },
  2030. /**
  2031. * 记住回复内容
  2032. */
  2033. async rememberReplyContent() {
  2034. const TAG = "记住回复内容 -- ";
  2035. let $formList = document.querySelectorAll("form");
  2036. if (!$formList.length) {
  2037. log.warn(TAG + "不存在表单");
  2038. return;
  2039. }
  2040. try {
  2041. await this.clearRelayHistoryRememberContentText();
  2042. } catch (error) {
  2043. log.error(error);
  2044. }
  2045. $formList.forEach(async ($form) => {
  2046. let $textarea = $form.querySelector("textarea");
  2047. let $replySubmit = $form.querySelector(`input[type="submit"]`);
  2048. if (!$textarea) {
  2049. return;
  2050. }
  2051. if (!$replySubmit) {
  2052. return;
  2053. }
  2054. log.success(`开始监听form --- 记住回复内容`, $form);
  2055. this.$data.db.get(this.$key.DB_KEY).then((result) => {
  2056. if (!result.data) {
  2057. return;
  2058. }
  2059. let localDataIndex = result.data.findIndex((item) => {
  2060. return this.checkUrlIsSame(window.location.href, item.url);
  2061. });
  2062. if (localDataIndex == -1) {
  2063. return;
  2064. }
  2065. let historyInputText = result.data[localDataIndex].text;
  2066. log.success("填入历史输入内容:" + historyInputText);
  2067. $textarea.value = historyInputText;
  2068. });
  2069. domUtils.on(
  2070. $textarea,
  2071. ["propertychange", "input"],
  2072. utils.debounce((event) => {
  2073. let data = {
  2074. url: window.location.href,
  2075. text: $textarea.value,
  2076. time: Date.now()
  2077. };
  2078. this.$data.db.get(this.$key.DB_KEY).then((result) => {
  2079. if (!result.success && result.event && result.event.type !== "success") {
  2080. log.warn(result);
  2081. return;
  2082. }
  2083. if (result.data == null) {
  2084. result.data = [];
  2085. }
  2086. let localDataIndex = result.data.findIndex((item) => {
  2087. return this.checkUrlIsSame(window.location.href, item.url);
  2088. });
  2089. if (localDataIndex !== -1) {
  2090. if (utils.isNull(data.text)) {
  2091. result.data.splice(localDataIndex, 1);
  2092. } else {
  2093. result.data[localDataIndex] = utils.assign(
  2094. result.data[localDataIndex],
  2095. data
  2096. );
  2097. }
  2098. } else {
  2099. result.data = result.data.concat(data);
  2100. }
  2101. this.$data.db.save(this.$key.DB_KEY, result.data).then((result2) => {
  2102. if (result2.success) ;
  2103. else {
  2104. log.error("保存失败", result2);
  2105. }
  2106. });
  2107. });
  2108. }, 25)
  2109. );
  2110. domUtils.on(
  2111. $form,
  2112. "submit",
  2113. (event) => {
  2114. log.info(`表单提交,刷新页面后清理内容:` + window.location.href);
  2115. _GM_setValue(
  2116. "delyClear_rememberReplyContent_url",
  2117. window.location.href
  2118. );
  2119. },
  2120. {
  2121. capture: true
  2122. }
  2123. );
  2124. });
  2125. },
  2126. /**
  2127. * 清理历史记住的回复内容
  2128. */
  2129. async clearRelayHistoryRememberContentText() {
  2130. const KEY2 = "delyClear_rememberReplyContent_url";
  2131. let delyClear_rememberReplyContent_url = _GM_getValue(KEY2);
  2132. if (delyClear_rememberReplyContent_url) {
  2133. let result = await this.$data.db.get(
  2134. this.$key.DB_KEY
  2135. );
  2136. if (!result.data) {
  2137. log.info("表单记录:数据库是空的");
  2138. return;
  2139. }
  2140. let localDataIndex = result.data.findIndex((item) => {
  2141. return this.checkUrlIsSame(
  2142. delyClear_rememberReplyContent_url,
  2143. item.url
  2144. );
  2145. });
  2146. if (localDataIndex == -1) {
  2147. log.info("表单记录:已不存在该数据");
  2148. _GM_deleteValue(KEY2);
  2149. return;
  2150. }
  2151. result.data.splice(localDataIndex, 1);
  2152. let saveResult = await this.$data.db.save(this.$key.DB_KEY, result.data);
  2153. if (saveResult.success) {
  2154. log.success("表单记录:成功清除");
  2155. _GM_deleteValue(KEY2);
  2156. } else {
  2157. log.error("表单记录:清除失败", result);
  2158. }
  2159. }
  2160. },
  2161. /**
  2162. * 检测两个url是否相同(不包括hash值)
  2163. * @param url1
  2164. * @param url2
  2165. */
  2166. checkUrlIsSame(url1, url2) {
  2167. let url1Obj = new URL(url1);
  2168. let url2Obj = new URL(url2);
  2169. return url1Obj.origin === url2Obj.origin && url1Obj.pathname === url2Obj.pathname;
  2170. },
  2171. /**
  2172. * 自动清理空间
  2173. * @param intervalDay 间隔天数
  2174. */
  2175. autoClearRememberReplayContent(intervalDay) {
  2176. const KEY2 = "gf-last-time-autoClearRememberReplayContent";
  2177. let lastClearTime = _GM_getValue(KEY2);
  2178. let intervalTime = intervalDay * 24 * 60 * 60 * 1e3;
  2179. if (lastClearTime) {
  2180. if (Date.now() - lastClearTime > intervalTime) {
  2181. _GM_setValue(KEY2, Date.now());
  2182. } else {
  2183. return;
  2184. }
  2185. }
  2186. _GM_setValue(KEY2, Date.now());
  2187. },
  2188. /**
  2189. * 获取所有的记住的回复内容
  2190. */
  2191. async getAllRememberReplyContent() {
  2192. let result = await this.$data.db.get(
  2193. this.$key.DB_KEY
  2194. );
  2195. return result.data ?? [];
  2196. },
  2197. /**
  2198. * 清空所有的记住的回复内容
  2199. */
  2200. async clearAllRememberReplyContent() {
  2201. let result = await this.$data.db.delete(this.$key.DB_KEY);
  2202. return result.success;
  2203. }
  2204. };
  2205. const PopsPanelUISetting = {
  2206. /**
  2207. * 面板-脚本列表|库
  2208. * @param type
  2209. * @param event
  2210. * @param rightHeaderElement
  2211. * @param rightContainerElement
  2212. * @returns
  2213. */
  2214. async UIScriptList(type, rightContainerElement) {
  2215. var _a2, _b, _c;
  2216. if (!GreasyforkMenu.isLogin) {
  2217. Qmsg.error(i18next.t("请先登录账号!"));
  2218. return;
  2219. }
  2220. let userLinkElement = GreasyforkMenu.getUserLinkElement();
  2221. let userLink = userLinkElement.href;
  2222. 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];
  2223. let loading = __pops.loading({
  2224. mask: {
  2225. enable: true
  2226. },
  2227. parent: rightContainerElement,
  2228. content: {
  2229. text: i18next.t("获取信息中,请稍后...")
  2230. },
  2231. addIndexCSS: false
  2232. });
  2233. let userInfo = await GreasyforkApi.getUserInfo(userId);
  2234. loading.close();
  2235. if (!userInfo) {
  2236. return;
  2237. }
  2238. log.info(userInfo);
  2239. let scriptList = type === "script-list" ? userInfo["scriptList"] : userInfo["scriptLibraryList"];
  2240. Qmsg.success(
  2241. i18next.t("获取成功,共 {{count}} 个", {
  2242. count: scriptList.length
  2243. })
  2244. );
  2245. for (const scriptInfo of scriptList) {
  2246. let liElement = domUtils.createElement("li", {
  2247. className: "w-script-list-item",
  2248. innerHTML: (
  2249. /*html*/
  2250. `
  2251. <div class="w-script-info">
  2252. <div class="w-script-name">
  2253. <a href="${scriptInfo["url"]}" target="_blank">${scriptInfo["name"]}</a>
  2254. </div>
  2255. <div class="w-script-fan-score">
  2256. <p>${i18next.t("评分:")}${scriptInfo["fan_score"]}</p>
  2257. </div>
  2258. <div class="w-script-locale">
  2259. <p>${i18next.t("语言:")}${scriptInfo["locale"]}</p>
  2260. </div>
  2261. <div class="w-script-version">
  2262. <p>${i18next.t("版本:")}${scriptInfo["version"]}</p>
  2263. </div>
  2264. <div class="w-script-update-time">
  2265. <p>${i18next.t("更新:")}${utils.getDaysDifference(
  2266. new Date(scriptInfo["code_updated_at"]).getTime(),
  2267. void 0,
  2268. "auto"
  2269. )}前</p>
  2270. </div>
  2271. </div>
  2272. `
  2273. )
  2274. });
  2275. let scriptInfoElement = liElement.querySelector(
  2276. ".w-script-info"
  2277. );
  2278. let buttonElement = domUtils.createElement("div", {
  2279. className: "pops-panel-button",
  2280. innerHTML: (
  2281. /*html*/
  2282. `
  2283. <button type="primary" data-icon="" data-righticon="false">
  2284. <span>${i18next.t("同步代码")}</span>
  2285. </button>
  2286. `
  2287. )
  2288. });
  2289. if (scriptInfo["deleted"]) {
  2290. liElement.classList.add("w-script-deleted");
  2291. buttonElement.querySelector("button").setAttribute("disabled", "true");
  2292. }
  2293. domUtils.on(buttonElement, "click", void 0, async function() {
  2294. log.success("同步", scriptInfo);
  2295. let btn = buttonElement.querySelector("button");
  2296. let span = buttonElement.querySelector(
  2297. "button span"
  2298. );
  2299. let iconElement = domUtils.createElement(
  2300. "i",
  2301. {
  2302. className: "pops-bottom-icon",
  2303. innerHTML: __pops.config.iconSVG.loading
  2304. },
  2305. {
  2306. "is-loading": true
  2307. }
  2308. );
  2309. btn.setAttribute("disabled", "true");
  2310. btn.setAttribute("data-icon", "true");
  2311. span.innerText = i18next.t("同步中...");
  2312. domUtils.before(span, iconElement);
  2313. let scriptId = scriptInfo == null ? void 0 : scriptInfo["id"];
  2314. let codeSyncFormData = await GreasyforkApi.getSourceCodeSyncFormData(
  2315. scriptId.toString()
  2316. );
  2317. if (codeSyncFormData) {
  2318. const SCRIPT_SYNC_TYPE_ID_FORMDATA_KEY = "script[script_sync_type_id]";
  2319. if (codeSyncFormData.has(SCRIPT_SYNC_TYPE_ID_FORMDATA_KEY)) {
  2320. let syncTypeId = codeSyncFormData.get(
  2321. SCRIPT_SYNC_TYPE_ID_FORMDATA_KEY
  2322. );
  2323. let syncMode = "";
  2324. if (syncTypeId.toString() === "1") {
  2325. syncMode = i18next.t("手动");
  2326. } else if (syncTypeId.toString() === "2") {
  2327. syncMode = i18next.t("自动");
  2328. } else if (syncTypeId.toString() === "3") {
  2329. syncMode = "webhook";
  2330. }
  2331. let oldSyncTypeElement = liElement.querySelector(
  2332. ".w-script-sync-type"
  2333. );
  2334. if (oldSyncTypeElement) {
  2335. oldSyncTypeElement.querySelector("p").innerText = i18next.t(
  2336. "同步方式:{{syncMode}}",
  2337. { syncMode }
  2338. );
  2339. } else {
  2340. domUtils.append(
  2341. scriptInfoElement,
  2342. /*html*/
  2343. `
  2344. <div class="w-script-sync-type">
  2345. <p>${i18next.t("同步方式:{{syncMode}}", {
  2346. syncMode
  2347. })}
  2348. </p>
  2349. </div>`
  2350. );
  2351. }
  2352. let syncUpdateResponse = await GreasyforkApi.sourceCodeSync(
  2353. scriptInfo["id"].toString(),
  2354. codeSyncFormData
  2355. );
  2356. if (syncUpdateResponse) {
  2357. Qmsg.success(i18next.t("同步成功"));
  2358. } else {
  2359. Qmsg.error(i18next.t("同步失败"));
  2360. }
  2361. } else {
  2362. Qmsg.error(i18next.t("该脚本未设置同步信息"));
  2363. }
  2364. }
  2365. btn.removeAttribute("disabled");
  2366. btn.removeAttribute("data-icon");
  2367. span.innerText = i18next.t("同步代码");
  2368. iconElement.remove();
  2369. });
  2370. liElement.appendChild(buttonElement);
  2371. rightContainerElement.appendChild(liElement);
  2372. }
  2373. }
  2374. };
  2375. const GreasyforkScriptsCode = {
  2376. init() {
  2377. PopsPanel.execMenuOnce("code-repairCodeLineNumber", () => {
  2378. this.repairCodeLineNumber();
  2379. });
  2380. PopsPanel.execMenuOnce("code-use-monaco-editor", () => {
  2381. this.coverEditorWithMonaco();
  2382. });
  2383. },
  2384. /**
  2385. * 修复代码的行号显示不够问题
  2386. * 超过1w行不会高亮代码
  2387. */
  2388. repairCodeLineNumber() {
  2389. log.info("修复代码的行号显示不够问题");
  2390. PopsPanel.execMenuOnce("beautifyGreasyforkBeautify", () => {
  2391. addStyle(
  2392. /*css*/
  2393. `
  2394. .code-container pre code .marker{
  2395. padding-left: 6px;
  2396. }
  2397. `
  2398. );
  2399. });
  2400. utils.waitNode(
  2401. "#script-content div.code-container pre.prettyprint ol"
  2402. ).then(($prettyPrintOL) => {
  2403. if ($prettyPrintOL.childElementCount >= 1e3) {
  2404. log.success(
  2405. `当前代码行数${$prettyPrintOL.childElementCount}行,超过1000行,优化行号显示问题`
  2406. );
  2407. addStyle(
  2408. /*css*/
  2409. `
  2410. pre.prettyprint{
  2411. padding-left: 26px;
  2412. }
  2413. `
  2414. );
  2415. }
  2416. });
  2417. },
  2418. /**
  2419. * 使用monacoEditor替换编辑器
  2420. */
  2421. coverEditorWithMonaco() {
  2422. log.info(`使用monacoEditor替换编辑器`);
  2423. addStyle(
  2424. /*css*/
  2425. `
  2426. @font-face {
  2427. font-family: 'codicon';
  2428. src: url('https://fastly.jsdelivr.net/npm/monaco-editor@0.52.0/min/vs/base/browser/ui/codicons/codicon/codicon.ttf') format('truetype');
  2429. }
  2430. `
  2431. );
  2432. addStyle(
  2433. /*css*/
  2434. `
  2435. .monaco-editor{
  2436. height: calc(100vh - 54px);
  2437. }
  2438. `
  2439. );
  2440. CommonUtil.addBlockCSS("#script-content .code-container > pre");
  2441. let $monacoScript = domUtils.createElement("script", {
  2442. type: "module",
  2443. innerHTML: `
  2444. import * as monaco from "https://fastly.jsdelivr.net/npm/monaco-editor@0.52.0/+esm";
  2445. window.monaco = monaco;
  2446. window.dispatchEvent(new CustomEvent("monaco-editor-ready"));
  2447. `
  2448. });
  2449. domUtils.append(document.head || document.documentElement, $monacoScript);
  2450. domUtils.append(document.head || document.documentElement, $monacoScript);
  2451. domUtils.on(
  2452. window,
  2453. "monaco-editor-ready",
  2454. async () => {
  2455. let monaco2 = _unsafeWindow.monaco;
  2456. let response = await httpx.get(window.location.href, { fetch: true });
  2457. if (!response.status) {
  2458. return;
  2459. }
  2460. let doc = domUtils.parseHTML(response.data.responseText, true, true);
  2461. let $originCodeContainer = doc.querySelector(
  2462. "#script-content .code-container > pre"
  2463. );
  2464. if (!$originCodeContainer) {
  2465. return;
  2466. }
  2467. let codeText = domUtils.text($originCodeContainer);
  2468. domUtils.ready(async () => {
  2469. let $codeContainer = await utils.waitNode(
  2470. "#script-content .code-container > pre",
  2471. 1e4
  2472. );
  2473. if (!$codeContainer) {
  2474. return;
  2475. }
  2476. let $monacoEditor = domUtils.createElement("div", {
  2477. className: "monaco-editor"
  2478. });
  2479. domUtils.after($codeContainer, $monacoEditor);
  2480. monaco2.editor.create($monacoEditor, {
  2481. value: codeText,
  2482. minimap: { enabled: true },
  2483. // 小地图
  2484. automaticLayout: true,
  2485. // 自动布局,
  2486. codeLens: true,
  2487. colorDecorators: true,
  2488. contextmenu: true,
  2489. readOnly: true,
  2490. //是否只读
  2491. formatOnPaste: true,
  2492. overviewRulerBorder: true,
  2493. // 滚动条的边框
  2494. scrollBeyondLastLine: true,
  2495. theme: "vs-dark",
  2496. // 主题
  2497. fontSize: window.innerWidth > 600 ? 14 : 12,
  2498. // 字体
  2499. wordWrap: "off",
  2500. // 换行
  2501. language: "javascript",
  2502. // 语言
  2503. folding: true,
  2504. // 是否启用代码折叠
  2505. foldingStrategy: "indentation"
  2506. // 代码可分小段折叠
  2507. });
  2508. });
  2509. },
  2510. { once: true }
  2511. );
  2512. }
  2513. };
  2514. 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';
  2515. const GreasyforkVersions = {
  2516. init() {
  2517. PopsPanel.execMenuOnce("beautifyHistoryVersionPage", () => {
  2518. return this.beautifyHistoryVersionPage();
  2519. });
  2520. PopsPanel.execMenuOnce("scripts-versions-addExtraTagButton", () => {
  2521. this.addExtraTagButton();
  2522. });
  2523. PopsPanel.execMenuOnce("scripts-versions-addCompareCodeButton", () => {
  2524. this.sourceDiffMonacoEditor();
  2525. });
  2526. },
  2527. /**
  2528. * 美化 历史版本 页面
  2529. */
  2530. beautifyHistoryVersionPage() {
  2531. log.info("美化 历史版本 页面");
  2532. let result = [];
  2533. result.push(addStyle(beautifyVersionsPageCSS));
  2534. result.push(
  2535. CommonUtil.addBlockCSS(
  2536. ".version-number",
  2537. ".version-date",
  2538. ".version-changelog"
  2539. )
  2540. );
  2541. domUtils.ready(function() {
  2542. let $historyVersion = document.querySelector(
  2543. "ul.history_versions"
  2544. );
  2545. if (!$historyVersion) {
  2546. Qmsg.error(i18next.t("未找到history_versions元素列表"));
  2547. return;
  2548. }
  2549. Array.from($historyVersion.children).forEach((liElement) => {
  2550. var _a2, _b;
  2551. let versionUrl = liElement.querySelector(".version-number a").href;
  2552. let versionNumber = liElement.querySelector(
  2553. ".version-number a"
  2554. ).innerText;
  2555. let versionDate = (_a2 = liElement.querySelector(".version-date")) == null ? void 0 : _a2.getAttribute("datetime");
  2556. let updateNote = ((_b = liElement.querySelector(".version-changelog")) == null ? void 0 : _b.innerHTML) || "";
  2557. let versionDateElement = domUtils.createElement("span", {
  2558. className: "script-version-date",
  2559. innerHTML: utils.formatTime(
  2560. versionDate,
  2561. i18next.t("yyyy年MM月dd日 HH:mm:ss")
  2562. )
  2563. });
  2564. let tagElement = domUtils.createElement("div", {
  2565. className: "script-tag",
  2566. innerHTML: (
  2567. /*html*/
  2568. `
  2569. <div class="script-tag-version">
  2570. <a href="${versionUrl}" class="flex-align-item-center">
  2571. <svg aria-label="Tag" role="img" height="16" viewBox="0 0 16 16" version="1.1" width="16">
  2572. <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>
  2573. </svg>
  2574. <span>${versionNumber}</span>
  2575. </a>
  2576. </div>`
  2577. )
  2578. });
  2579. let boxBodyElement = domUtils.createElement("div", {
  2580. className: "script-note-box-body",
  2581. innerHTML: updateNote
  2582. });
  2583. liElement.appendChild(versionDateElement);
  2584. liElement.appendChild(tagElement);
  2585. liElement.appendChild(boxBodyElement);
  2586. });
  2587. });
  2588. return result;
  2589. },
  2590. /**
  2591. * 添加额外的标签按钮
  2592. */
  2593. addExtraTagButton() {
  2594. log.info("添加额外的标签按钮");
  2595. domUtils.ready(() => {
  2596. document.querySelectorAll(".script-tag-version").forEach(($tagVersion) => {
  2597. var _a2, _b;
  2598. let $anchor = $tagVersion.querySelector("a");
  2599. if (!$anchor) {
  2600. return;
  2601. }
  2602. let urlObj = new URL($anchor.href);
  2603. let scriptId = (_a2 = urlObj.pathname.match(/\/scripts\/([\d]+)/)) == null ? void 0 : _a2[1];
  2604. let scriptVersion = urlObj.searchParams.get("version");
  2605. let scriptName = (_b = urlObj.pathname.match(/\/scripts\/[\d]+-(.+)/)) == null ? void 0 : _b[1];
  2606. let installUrl = GreasyforkUrlUtils.getInstallUrl(
  2607. scriptId,
  2608. scriptVersion,
  2609. scriptName
  2610. );
  2611. let codeUrl = GreasyforkUrlUtils.getCodeUrl(scriptId, scriptVersion);
  2612. let $buttonTag = domUtils.createElement("div", {
  2613. className: "scripts-tag-install",
  2614. innerHTML: (
  2615. /*html*/
  2616. `
  2617. <a class="script-btn-install install-link" data-install-format="js" target="_blank" href="${installUrl}">${i18next.t(
  2618. "安装此脚本"
  2619. )}</a>
  2620. <a class="script-btn-see-code" target="_blank" href="${codeUrl}">${i18next.t(
  2621. "查看代码"
  2622. )}</a>
  2623. `
  2624. )
  2625. });
  2626. domUtils.after($tagVersion, $buttonTag);
  2627. });
  2628. });
  2629. },
  2630. /**
  2631. * 源码对比(monacoEditor)
  2632. */
  2633. sourceDiffMonacoEditor() {
  2634. log.info(`源码对比(monacoEditor)`);
  2635. addStyle(
  2636. /*css*/
  2637. `
  2638. @font-face {
  2639. font-family: 'codicon';
  2640. src: url('https://fastly.jsdelivr.net/npm/monaco-editor@0.52.0/min/vs/base/browser/ui/codicons/codicon/codicon.ttf') format('truetype');
  2641. }
  2642. `
  2643. );
  2644. let $monacoScript = domUtils.createElement("script", {
  2645. type: "module",
  2646. innerHTML: `
  2647. import * as monaco from "https://fastly.jsdelivr.net/npm/monaco-editor@0.52.0/+esm";
  2648. window.monaco = monaco;
  2649. window.dispatchEvent(new CustomEvent("monaco-editor-ready"));
  2650. `
  2651. });
  2652. domUtils.append(document.head || document.documentElement, $monacoScript);
  2653. domUtils.on(
  2654. window,
  2655. "monaco-editor-ready",
  2656. () => {
  2657. let monaco2 = unsafeWindow.monaco;
  2658. domUtils.ready(() => {
  2659. $$(
  2660. `#script-content form[action*="/diff"] input[type="submit"]`
  2661. ).forEach(($submit) => {
  2662. let $compareButton = domUtils.createElement(
  2663. "input",
  2664. {
  2665. type: "button",
  2666. value: i18next.t("对比选中版本差异(monacoEditor)")
  2667. },
  2668. {
  2669. style: "margin-left: 10px;"
  2670. }
  2671. );
  2672. domUtils.after($submit, $compareButton);
  2673. domUtils.on($compareButton, "click", async (event) => {
  2674. utils.preventEvent(event);
  2675. let $form = $submit.closest("form");
  2676. let formData = new FormData($form);
  2677. let compareLeftVersion = formData.get("v1");
  2678. let compareRighttVersion = formData.get("v2");
  2679. if (compareLeftVersion === compareRighttVersion) {
  2680. Qmsg.warning(i18next.t("版本号相同,不需要比较源码"));
  2681. return;
  2682. }
  2683. let loading = Qmsg.loading(i18next.t("正在获取对比文本中..."));
  2684. let scriptId = GreasyforkUrlUtils.getScriptId();
  2685. let response = await httpx.get(
  2686. `https://greasyfork.org/zh-CN/scripts/${scriptId}.json`,
  2687. {
  2688. fetch: true,
  2689. responseType: "json"
  2690. }
  2691. );
  2692. if (!response.status) {
  2693. loading.close();
  2694. return;
  2695. }
  2696. let respJSON = utils.toJSON(response.data.responseText);
  2697. let code_url = respJSON["code_url"];
  2698. let compareLeftUrl = code_url.replace(
  2699. `/${scriptId}`,
  2700. `/${scriptId}/${compareLeftVersion}`
  2701. );
  2702. let compareRightUrl = code_url.replace(
  2703. `/${scriptId}`,
  2704. `/${scriptId}/${compareRighttVersion}`
  2705. );
  2706. let compareLeftText = "";
  2707. let compareRightText = "";
  2708. let compareLeftResponse = await httpx.get(compareLeftUrl);
  2709. if (!compareLeftResponse.status) {
  2710. loading.close();
  2711. return;
  2712. }
  2713. compareLeftText = compareLeftResponse.data.responseText;
  2714. let compareRightResponse = await httpx.get(compareRightUrl);
  2715. if (!compareRightResponse.status) {
  2716. loading.close();
  2717. return;
  2718. }
  2719. compareRightText = compareRightResponse.data.responseText;
  2720. loading.close();
  2721. let { recovery } = CommonUtil.lockScroll();
  2722. let $alert = __pops.alert({
  2723. title: {
  2724. text: i18next.t("代码对比"),
  2725. html: false,
  2726. position: "center"
  2727. },
  2728. content: {
  2729. html: true,
  2730. text: (
  2731. /*html*/
  2732. `
  2733. <div class="monaco-editor-diff-container">
  2734. <div class="monaco-editor-diff"></div>
  2735. </div>
  2736. `
  2737. )
  2738. },
  2739. mask: {
  2740. enable: true,
  2741. clickEvent: {
  2742. toClose: false,
  2743. toHide: false
  2744. }
  2745. },
  2746. btn: {
  2747. ok: {
  2748. enable: false
  2749. },
  2750. close: {
  2751. callback(details, event2) {
  2752. details.close();
  2753. recovery();
  2754. }
  2755. }
  2756. },
  2757. zIndex() {
  2758. let maxZIndex = utils.getMaxZIndex();
  2759. let popsMaxZIndex = __pops.config.InstanceUtils.getPopsMaxZIndex().zIndex;
  2760. return utils.getMaxValue(maxZIndex, popsMaxZIndex) + 100;
  2761. },
  2762. useShadowRoot: false,
  2763. width: "90vw",
  2764. height: "90vh",
  2765. drag: true,
  2766. style: (
  2767. /*css*/
  2768. `
  2769. .monaco-editor-diff-container{
  2770. width: 100%;
  2771. height: 100%;
  2772. }
  2773. .monaco-editor-diff{
  2774. width: 100%;
  2775. height: 100%;
  2776. }
  2777. .pops[type-value="alert"] .pops-alert-title{
  2778. --container-title-height: 40px;
  2779. }
  2780. `
  2781. )
  2782. });
  2783. $alert.$shadowRoot.querySelector(
  2784. ".monaco-editor-diff-container"
  2785. );
  2786. let $monacoEditor = $alert.$shadowRoot.querySelector(
  2787. ".monaco-editor-diff"
  2788. );
  2789. let monacoEditor = monaco2.editor.createDiffEditor($monacoEditor, {
  2790. hideUnchangedRegions: {
  2791. enabled: true
  2792. },
  2793. minimap: { enabled: true },
  2794. // 小地图
  2795. automaticLayout: true,
  2796. // 自动布局,
  2797. codeLens: true,
  2798. colorDecorators: true,
  2799. contextmenu: false,
  2800. readOnly: true,
  2801. //是否只读
  2802. formatOnPaste: true,
  2803. overviewRulerBorder: true,
  2804. // 滚动条的边框
  2805. scrollBeyondLastLine: true,
  2806. theme: "vs-dark",
  2807. // 主题
  2808. fontSize: window.innerWidth > 600 ? 14 : 12,
  2809. // 字体
  2810. wordWrap: "off",
  2811. // 换行
  2812. language: "javascript"
  2813. // 语言
  2814. });
  2815. const originModel = monaco2.editor.createModel(
  2816. compareLeftText,
  2817. "javascript"
  2818. );
  2819. const modifyModel = monaco2.editor.createModel(
  2820. compareRightText,
  2821. "javascript"
  2822. );
  2823. monacoEditor.setModel({
  2824. original: originModel,
  2825. modified: modifyModel
  2826. });
  2827. });
  2828. });
  2829. });
  2830. },
  2831. { once: true }
  2832. );
  2833. }
  2834. };
  2835. const PanelUISize = {
  2836. /**
  2837. * 一般设置界面的尺寸
  2838. */
  2839. setting: {
  2840. get width() {
  2841. return window.innerWidth < 550 ? "88vw" : "550px";
  2842. },
  2843. get height() {
  2844. return window.innerHeight < 450 ? "70vh" : "450px";
  2845. }
  2846. },
  2847. /**
  2848. * 功能丰富,aside铺满了的设置界面,要稍微大一点
  2849. */
  2850. settingBig: {
  2851. get width() {
  2852. return window.innerWidth < 800 ? "92vw" : "800px";
  2853. },
  2854. get height() {
  2855. return window.innerHeight < 600 ? "80vh" : "600px";
  2856. }
  2857. },
  2858. /**
  2859. * 信息界面,一般用于提示信息之类
  2860. */
  2861. info: {
  2862. get width() {
  2863. return window.innerWidth < 350 ? "350px" : "350px";
  2864. },
  2865. get height() {
  2866. return window.innerHeight < 250 ? "250px" : "250px";
  2867. }
  2868. }
  2869. };
  2870. let userCollection = [];
  2871. const GreasyforkScriptsCollectEvent = async function(scriptId) {
  2872. log.info("当前脚本id:" + scriptId);
  2873. if (!GreasyforkMenu.isLogin) {
  2874. log.error("请先登录账号");
  2875. Qmsg.error(i18next.t("请先登录账号"));
  2876. return;
  2877. }
  2878. let userId = GreasyforkUrlUtils.getUserId(
  2879. GreasyforkMenu.getUserLinkElement().href
  2880. );
  2881. if (userId == null) {
  2882. log.error("获取用户id失败");
  2883. Qmsg.error(i18next.t("获取用户id失败"));
  2884. return;
  2885. }
  2886. if (!userCollection.length) {
  2887. let loading = Qmsg.loading(i18next.t("获取收藏夹中..."));
  2888. userCollection = await GreasyforkApi.getUserCollection(userId) || [];
  2889. loading.close();
  2890. if (!userCollection.length) {
  2891. return;
  2892. }
  2893. }
  2894. let alertHTML = "";
  2895. const checkFavoriteFormInfo = (form, scriptId2) => {
  2896. let flag = false;
  2897. scriptId2 = scriptId2.toString().trim();
  2898. for (const [key, value] of form.entries()) {
  2899. if (key === "scripts-included[]" && value.toString().trim() === scriptId2) {
  2900. flag = true;
  2901. break;
  2902. }
  2903. }
  2904. return flag;
  2905. };
  2906. userCollection.forEach((userCollectInfo) => {
  2907. alertHTML += /*html*/
  2908. `
  2909. <li class="user-collect-item" data-id="${userCollectInfo.id}" data-name="${userCollectInfo.name}">
  2910. <div class="user-collect-name">${userCollectInfo.name}</div>
  2911. <div class="user-collect-btn-container">
  2912. <div class="pops-panel-button collect-add-script-id">
  2913. <button type="primary" data-icon="" data-righticon="">
  2914. <span>${i18next.t("添加")}</span>
  2915. </button>
  2916. </div>
  2917. <div class="pops-panel-button collect-delete-script-id">
  2918. <button type="danger" data-icon="" data-righticon="">
  2919. <span>${i18next.t("刪除")}</span>
  2920. </button>
  2921. </div>
  2922. </div>
  2923. </li>
  2924. `;
  2925. });
  2926. let collectionDialog = __pops.alert({
  2927. title: {
  2928. text: i18next.t("收藏集"),
  2929. position: "center"
  2930. },
  2931. content: {
  2932. html: true,
  2933. text: (
  2934. /*html*/
  2935. `<ul>${alertHTML}</ul>`
  2936. )
  2937. },
  2938. mask: {
  2939. enable: true,
  2940. clickEvent: {
  2941. toClose: true
  2942. }
  2943. },
  2944. btn: {
  2945. ok: {
  2946. enable: false
  2947. }
  2948. },
  2949. width: __pops.isPhone() ? "92vw" : "500px",
  2950. height: "auto",
  2951. drag: true,
  2952. only: true,
  2953. style: (
  2954. /*css*/
  2955. `
  2956. .pops{
  2957. --content-max-height: 400px;
  2958. max-height: var(--content-max-height);
  2959. }
  2960. .pops[type-value=alert] .pops-alert-content {
  2961. max-height: calc(var(--content-max-height) - var(--container-title-height) - var(--container-bottom-btn-height));
  2962. }
  2963. .user-collect-item{
  2964. -webkit-user-select: none;
  2965. user-select: none;
  2966. padding: 5px 10px;
  2967. display: flex;
  2968. align-items: center;
  2969. justify-content: space-between;
  2970. border-bottom: 1px dotted #c9c9c9;
  2971. }
  2972. .user-collect-name{
  2973.  
  2974. }
  2975. .user-collect-item:hover{
  2976. }
  2977. .user-collect-btn-container{
  2978. margin-left: 10px;
  2979. display: flex;
  2980. }
  2981. `
  2982. )
  2983. });
  2984. domUtils.on(
  2985. collectionDialog.$shadowRoot,
  2986. "click",
  2987. ".collect-add-script-id",
  2988. async function(event) {
  2989. let $userCollectItem = event.target.closest(
  2990. ".user-collect-item"
  2991. );
  2992. let setsId = $userCollectItem.dataset.id;
  2993. $userCollectItem.dataset.name;
  2994. let loading = Qmsg.loading(i18next.t("添加中..."));
  2995. try {
  2996. let formData = await GreasyforkApi.getUserCollectionInfo(
  2997. userId,
  2998. setsId
  2999. );
  3000. if (!formData) {
  3001. return;
  3002. }
  3003. if (checkFavoriteFormInfo(formData, scriptId)) {
  3004. Qmsg.warning(i18next.t("该脚本已经在该收藏集中"));
  3005. return;
  3006. }
  3007. let editForm = utils.cloneFormData(formData);
  3008. let saveEditForm = utils.cloneFormData(formData);
  3009. editForm.set("add-script", scriptId.toString());
  3010. editForm.set("script-action", "i");
  3011. saveEditForm.append("scripts-included[]", scriptId.toString());
  3012. saveEditForm.set("save", "1");
  3013. let addFormDataSearchParams = new URLSearchParams(editForm);
  3014. let saveFormDataSearchParams = new URLSearchParams(saveEditForm);
  3015. let addData = Array.from(addFormDataSearchParams).map(
  3016. // @ts-ignore
  3017. ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
  3018. ).join("&");
  3019. let saveData = Array.from(saveFormDataSearchParams).map(
  3020. // @ts-ignore
  3021. ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
  3022. ).join("&");
  3023. log.info("添加的数据", addData);
  3024. log.info("保存的数据", saveData);
  3025. let changeResultDoc = await GreasyforkApi.updateUserSetsInfo(
  3026. userId,
  3027. setsId,
  3028. addData
  3029. );
  3030. if (!changeResultDoc) {
  3031. return;
  3032. }
  3033. let $changeScriptSet = changeResultDoc.querySelector(".change-script-set");
  3034. if (!$changeScriptSet) {
  3035. Qmsg.error(
  3036. i18next.t("添加失败,{{selector}}元素不存在", {
  3037. selector: ".change-script-set"
  3038. })
  3039. );
  3040. return;
  3041. }
  3042. let $section = $changeScriptSet.querySelector("section");
  3043. if (!$section) {
  3044. Qmsg.error(
  3045. i18next.t("添加失败,{{selector}}元素不存在", {
  3046. selector: "section"
  3047. })
  3048. );
  3049. return;
  3050. }
  3051. let $alertElement = $section.querySelector(".alert");
  3052. if ($alertElement) {
  3053. __pops.alert({
  3054. title: {
  3055. text: i18next.t("添加失败"),
  3056. position: "center"
  3057. },
  3058. content: {
  3059. text: $alertElement.innerHTML,
  3060. html: true
  3061. },
  3062. mask: {
  3063. enable: true,
  3064. clickEvent: {
  3065. toClose: true
  3066. }
  3067. },
  3068. style: (
  3069. /*css*/
  3070. `
  3071. .pops-alert-content{
  3072. font-style: italic;
  3073. background-color: #ffc;
  3074. border: none;
  3075. border-left: 6px solid #FFEB3B;
  3076. padding: .5em;
  3077. }
  3078. `
  3079. ),
  3080. drag: true,
  3081. dragLimit: true,
  3082. width: PanelUISize.info.width,
  3083. height: PanelUISize.info.height
  3084. });
  3085. return;
  3086. }
  3087. let changeScriptForm = new FormData($changeScriptSet);
  3088. let changeFlag = checkFavoriteFormInfo(changeScriptForm, scriptId);
  3089. if (!changeFlag) {
  3090. log.error("添加失败,提交的添加请求中不包含该脚本id");
  3091. Qmsg.error(i18next.t("添加失败,表单数据中不包含该脚本"));
  3092. return;
  3093. }
  3094. await GreasyforkApi.updateUserSetsInfo(userId, setsId, saveData);
  3095. Qmsg.success(i18next.t("添加成功"));
  3096. } catch (error) {
  3097. console.error(error);
  3098. } finally {
  3099. loading.close();
  3100. }
  3101. }
  3102. );
  3103. domUtils.on(
  3104. collectionDialog.$shadowRoot,
  3105. "click",
  3106. ".collect-delete-script-id",
  3107. async function(event) {
  3108. let $collectItem = event.target.closest(
  3109. ".user-collect-item"
  3110. );
  3111. let setsId = $collectItem.dataset.id;
  3112. $collectItem.dataset.name;
  3113. let loading = Qmsg.loading(i18next.t("删除中..."));
  3114. try {
  3115. let formData = await GreasyforkApi.getUserCollectionInfo(
  3116. userId,
  3117. setsId
  3118. );
  3119. if (!formData) {
  3120. return;
  3121. }
  3122. if (!checkFavoriteFormInfo(formData, scriptId)) {
  3123. Qmsg.info(
  3124. i18next.t("已删除:{{scriptId}}", {
  3125. scriptId
  3126. })
  3127. );
  3128. return;
  3129. }
  3130. let editForm = utils.cloneFormData(formData, (key, value) => {
  3131. return key === "scripts-included[]" && typeof value === "string" && value.toString().trim() === scriptId.toString().trim();
  3132. });
  3133. let saveEditForm = utils.cloneFormData(editForm);
  3134. editForm.set("remove-scripts-included[]", scriptId.toString());
  3135. editForm.set("remove-selected-scripts", "i");
  3136. editForm.delete("script-action");
  3137. saveEditForm.set("save", "1");
  3138. let deleteFormDataSearchParams = new URLSearchParams(editForm);
  3139. let saveFormDataSearchParams = new URLSearchParams(saveEditForm);
  3140. let removeData = Array.from(deleteFormDataSearchParams).map(
  3141. // @ts-ignore
  3142. ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
  3143. ).join("&");
  3144. let saveData = Array.from(saveFormDataSearchParams).map(
  3145. // @ts-ignore
  3146. ([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`
  3147. ).join("&");
  3148. log.info("删除的数据", removeData);
  3149. log.info("保存的数据", saveData);
  3150. let changeResultDoc = await GreasyforkApi.updateUserSetsInfo(
  3151. userId,
  3152. setsId,
  3153. removeData
  3154. );
  3155. if (!changeResultDoc) {
  3156. return;
  3157. }
  3158. let $changeScriptSet = changeResultDoc.querySelector(".change-script-set");
  3159. if (!$changeScriptSet) {
  3160. Qmsg.error(
  3161. i18next.t("删除失败,{{selector}}元素不存在", {
  3162. selector: ".change-script-set"
  3163. })
  3164. );
  3165. return;
  3166. }
  3167. let changeScriptForm = new FormData($changeScriptSet);
  3168. let changeFlag = checkFavoriteFormInfo(changeScriptForm, scriptId);
  3169. if (changeFlag) {
  3170. log.error("删除失败,提交的删除请求中包含该脚本id");
  3171. Qmsg.error(i18next.t("删除失败,表单数据中仍包含该脚本"));
  3172. return;
  3173. }
  3174. await GreasyforkApi.updateUserSetsInfo(userId, setsId, saveData);
  3175. Qmsg.success(i18next.t("删除成功"));
  3176. } catch (error) {
  3177. console.error(error);
  3178. } finally {
  3179. loading.close();
  3180. }
  3181. }
  3182. );
  3183. };
  3184. const GreasyforkScripts = {
  3185. init() {
  3186. if (GreasyforkRouter.isCode()) {
  3187. GreasyforkScriptsCode.init();
  3188. } else if (GreasyforkRouter.isVersion()) {
  3189. GreasyforkVersions.init();
  3190. }
  3191. if (GreasyforkRouter.isCodeStrict()) {
  3192. PopsPanel.execMenuOnce("fullScreenOptimization", () => {
  3193. this.fullScreenOptimization();
  3194. });
  3195. PopsPanel.execMenuOnce("addCopyCodeButton", () => {
  3196. this.addCopyCodeButton();
  3197. });
  3198. }
  3199. PopsPanel.execMenuOnce("addCollectionButton", () => {
  3200. this.addCollectionButton();
  3201. });
  3202. PopsPanel.execMenuOnce("addFindReferenceButton", () => {
  3203. this.setFindCodeSearchBtn();
  3204. });
  3205. domUtils.ready(() => {
  3206. PopsPanel.execMenuOnce("scriptHomepageAddedTodaySUpdate", () => {
  3207. this.scriptHomepageAddedTodaySUpdate();
  3208. });
  3209. });
  3210. },
  3211. /**
  3212. * 添加【收藏】按钮
  3213. */
  3214. addCollectionButton() {
  3215. log.info("添加收藏按钮");
  3216. utils.waitNode("ul#script-links li.current span").then(() => {
  3217. let $collectBtn = domUtils.createElement("li", {
  3218. innerHTML: `
  3219. <a href="javascript:;">
  3220. <span>${i18next.t("收藏")}</span>
  3221. </a>`
  3222. });
  3223. domUtils.append(
  3224. document.querySelector("ul#script-links"),
  3225. $collectBtn
  3226. );
  3227. domUtils.on($collectBtn, "click", () => {
  3228. let scriptIdMatch = window.location.pathname.match(/scripts\/([\d]+)/i);
  3229. if (!scriptIdMatch) {
  3230. log.error(scriptIdMatch, window.location.pathname);
  3231. Qmsg.error(i18next.t("获取脚本id失败"));
  3232. return;
  3233. }
  3234. let scriptId = scriptIdMatch[scriptIdMatch.length - 1];
  3235. GreasyforkScriptsCollectEvent(scriptId);
  3236. });
  3237. });
  3238. },
  3239. /**
  3240. * F11全屏,F键代码全屏
  3241. */
  3242. fullScreenOptimization() {
  3243. log.info("F11全屏,F键代码全屏");
  3244. addStyle(
  3245. /*css*/
  3246. `
  3247. .code-container:has(.code-wide-screen){
  3248. height: auto !important;
  3249. }
  3250. .code-wide-screen{
  3251. position: absolute !important;
  3252. top: 0 !important;
  3253. left: 0 !important;
  3254. right: 0 !important;
  3255. bottom: 0 !important;
  3256. margin: 0 !important;
  3257. padding: 0 !important;
  3258. width: 100% !important;
  3259. height: 100% !important;
  3260. min-width: 100% !important;
  3261. min-height: 100% !important;
  3262. max-width: 100% !important;
  3263. max-height: 100% !important;
  3264. z-index: 1000000 !important;
  3265. }
  3266. `
  3267. );
  3268. let isFullScreen = false;
  3269. domUtils.keydown(
  3270. _unsafeWindow,
  3271. function(event) {
  3272. if (event.key.toLowerCase() === "f") {
  3273. let $code = $(".monaco-editor") || $("#script-content div.code-container code");
  3274. if (event.altKey && event.shiftKey) {
  3275. log.info(`宽屏`);
  3276. utils.preventEvent(event);
  3277. if ($code.classList.contains("code-wide-screen")) {
  3278. $code.classList.remove("code-wide-screen");
  3279. } else {
  3280. $code.classList.add("code-wide-screen");
  3281. }
  3282. } else if (!event.altKey && !event.ctrlKey && !event.shiftKey && !event.metaKey) {
  3283. log.info(`全屏`);
  3284. utils.preventEvent(event);
  3285. if (isFullScreen) {
  3286. utils.exitFullScreen($code);
  3287. isFullScreen = false;
  3288. } else {
  3289. utils.enterFullScreen($code);
  3290. isFullScreen = true;
  3291. }
  3292. }
  3293. }
  3294. },
  3295. {
  3296. capture: true
  3297. }
  3298. );
  3299. },
  3300. /**
  3301. * 设置代码搜索按钮(对于库)
  3302. */
  3303. setFindCodeSearchBtn() {
  3304. log.info("设置代码搜索按钮(对于库)");
  3305. utils.waitNode("ul#script-links li.current span").then(() => {
  3306. let searchBtn = domUtils.createElement("li", {
  3307. innerHTML: `
  3308. <a href="javascript:;">
  3309. <span>${i18next.t("寻找引用")}</span>
  3310. </a>`
  3311. });
  3312. domUtils.append(
  3313. document.querySelector(
  3314. "ul#script-links"
  3315. ),
  3316. searchBtn
  3317. );
  3318. domUtils.on(searchBtn, "click", async function() {
  3319. let scriptIdMatch = window.location.pathname.match(/scripts\/([\d]+)/i);
  3320. if (!scriptIdMatch) {
  3321. log.error(scriptIdMatch, window.location.pathname);
  3322. Qmsg.error(i18next.t("获取脚本id失败"));
  3323. return;
  3324. }
  3325. let scriptId = scriptIdMatch[scriptIdMatch.length - 1];
  3326. window.location.href = GreasyforkUrlUtils.getCodeSearchUrl(
  3327. `greasyfork.org/scripts/${scriptId}`
  3328. );
  3329. });
  3330. });
  3331. },
  3332. /**
  3333. * 脚本首页新增【今日检查】
  3334. */
  3335. async scriptHomepageAddedTodaySUpdate() {
  3336. if (!document.querySelector("#install-area")) {
  3337. return;
  3338. }
  3339. log.info("脚本首页新增【今日检查】");
  3340. let scriptStatsJSON = await GreasyforkApi.getScriptStats(
  3341. GreasyforkUrlUtils.getScriptId()
  3342. );
  3343. if (!scriptStatsJSON) {
  3344. return;
  3345. }
  3346. log.info("统计信息", scriptStatsJSON);
  3347. let todayStatsJSON = scriptStatsJSON[utils.formatTime(void 0, "yyyy-MM-dd")];
  3348. if (!todayStatsJSON) {
  3349. log.error("今日份的统计信息不存在");
  3350. return;
  3351. }
  3352. let update_checks = todayStatsJSON["update_checks"];
  3353. log.info("今日统计信息", todayStatsJSON);
  3354. domUtils.after(
  3355. "dd.script-show-daily-installs",
  3356. domUtils.createElement("dt", {
  3357. className: "script-show-daily-update_checks",
  3358. innerHTML: `<span>${i18next.t("今日检查")}</span>`
  3359. })
  3360. );
  3361. domUtils.after(
  3362. "dt.script-show-daily-update_checks",
  3363. domUtils.createElement("dd", {
  3364. className: "script-show-daily-update_checks",
  3365. innerHTML: "<span>" + update_checks + "</span>"
  3366. })
  3367. );
  3368. },
  3369. /**
  3370. * 添加复制代码按钮
  3371. */
  3372. addCopyCodeButton() {
  3373. log.info("添加复制代码按钮");
  3374. utils.waitNode("div#script-content div.code-container").then(($codeContainer) => {
  3375. let copyButton = domUtils.createElement(
  3376. "button",
  3377. {
  3378. textContent: i18next.t("复制代码")
  3379. },
  3380. {
  3381. style: "margin-bottom: 1em;"
  3382. }
  3383. );
  3384. domUtils.on(copyButton, "click", async function() {
  3385. let loading = Qmsg.loading(i18next.t("加载文件中..."));
  3386. let getResp = await httpx.get(
  3387. `https://greasyfork.org/zh-CN/scripts/${GreasyforkUrlUtils.getScriptId()}.json`,
  3388. {
  3389. fetch: true,
  3390. responseType: "json"
  3391. }
  3392. );
  3393. if (!getResp.status) {
  3394. loading.close();
  3395. return;
  3396. }
  3397. let respJSON = utils.toJSON(getResp.data.responseText);
  3398. let code_url = respJSON["code_url"];
  3399. log.success("代码地址:", code_url);
  3400. let scriptJS = await httpx.get(code_url);
  3401. if (!scriptJS.status) {
  3402. loading.close();
  3403. return;
  3404. }
  3405. loading.close();
  3406. utils.setClip(scriptJS.data.responseText);
  3407. Qmsg.success(i18next.t("复制成功"));
  3408. });
  3409. domUtils.before($codeContainer, copyButton);
  3410. });
  3411. }
  3412. };
  3413. 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.install-link {\r\n border-radius: 0.25rem 0.25rem 0.25rem 0.25rem;\r\n}\r\n.install-link:has(+ .install-help-link) {\r\n border-radius: 0.25rem 0 0 0.25rem;\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';
  3414. const GreasyforkCheckVersion = {
  3415. /** 获取 TamperMonkey 暴露在window下的函数 */
  3416. getTampermonkey: () => {
  3417. var _a2;
  3418. return (_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Tampermonkey;
  3419. },
  3420. /** 获取 Violentmonkey 暴露在window下的函数 */
  3421. getViolentmonkey: () => {
  3422. var _a2;
  3423. return (_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Violentmonkey;
  3424. },
  3425. /** 获取 ScriptCat 暴露在window下的函数 */
  3426. getScriptCat: () => {
  3427. var _a2;
  3428. return (_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Scriptcat;
  3429. },
  3430. /**
  3431. * 获取脚本容器启用状态
  3432. */
  3433. getScriptContainerStatus() {
  3434. var _a2, _b, _c;
  3435. let containerStatus = {
  3436. Tampermonkey: false,
  3437. Violentmonkey: false,
  3438. ScriptCat: false
  3439. };
  3440. if ((_a2 = _unsafeWindow.external) == null ? void 0 : _a2.Tampermonkey) {
  3441. containerStatus.Tampermonkey = true;
  3442. }
  3443. if ((_b = _unsafeWindow.external) == null ? void 0 : _b.Violentmonkey) {
  3444. containerStatus.Violentmonkey = true;
  3445. }
  3446. if ((_c = _unsafeWindow.external) == null ? void 0 : _c.Scriptcat) {
  3447. containerStatus.ScriptCat = true;
  3448. }
  3449. return containerStatus;
  3450. },
  3451. /**
  3452. * 获取已注册的脚本容器名
  3453. */
  3454. getRegisterScriptContainerNameList() {
  3455. let allScriptContainerStatus = this.getScriptContainerStatus();
  3456. let isRegisterScriptContainer = allScriptContainerStatus;
  3457. let scriptContainerNameList = [];
  3458. Object.keys(isRegisterScriptContainer).forEach((containerName) => {
  3459. let containerEnable = Reflect.get(
  3460. isRegisterScriptContainer,
  3461. containerName
  3462. );
  3463. if (containerEnable) {
  3464. scriptContainerNameList.push(containerName);
  3465. }
  3466. });
  3467. return scriptContainerNameList;
  3468. },
  3469. /**
  3470. * 获取脚本安装的版本号
  3471. * @param name 脚本名
  3472. * @param namespace 脚本命名空间
  3473. */
  3474. getInstalledVersion(name, namespace) {
  3475. return new Promise((resolve, reject) => {
  3476. const tm = this.getTampermonkey();
  3477. if (tm) {
  3478. tm.isInstalled(name, namespace, function(data) {
  3479. if (data.installed) {
  3480. resolve(data.version);
  3481. } else {
  3482. resolve(null);
  3483. }
  3484. });
  3485. return;
  3486. }
  3487. const vm = this.getViolentmonkey();
  3488. if (vm) {
  3489. vm.isInstalled(name, namespace).then(resolve);
  3490. return;
  3491. }
  3492. const scriptCat = this.getScriptCat();
  3493. if (scriptCat) {
  3494. scriptCat.isInstalled(name, namespace, function(data) {
  3495. if (data.installed) {
  3496. resolve(data.version);
  3497. } else {
  3498. resolve(null);
  3499. }
  3500. });
  3501. return;
  3502. }
  3503. reject(new TypeError("获取脚本容器暴露的external信息失败"));
  3504. });
  3505. },
  3506. /**
  3507. * https://developer.mozilla.org/en/docs/Toolkit_version_format
  3508. *
  3509. * 比较版本号
  3510. * @param a 版本号
  3511. * @param b 版本号
  3512. * @returns
  3513. * + -1 该版本号低
  3514. * + 0 该版本号和比较的版本号相同
  3515. * + 1 该版本号高
  3516. */
  3517. compareVersions(a, b) {
  3518. if (a === b) {
  3519. return 0;
  3520. }
  3521. const aParts = a.split(".");
  3522. const bParts = b.split(".");
  3523. for (let i = 0; i < aParts.length; i++) {
  3524. const result = this.compareVersionPart(aParts[i], bParts[i]);
  3525. if (result !== 0) {
  3526. return result;
  3527. }
  3528. }
  3529. return 0;
  3530. },
  3531. compareVersionPart(partA, partB) {
  3532. const partAParts = this.parseVersionPart(partA);
  3533. const partBParts = this.parseVersionPart(partB);
  3534. for (let i = 0; i < partAParts.length; i++) {
  3535. if (partAParts[i].length > 0 && partBParts[i].length === 0) {
  3536. return -1;
  3537. }
  3538. if (partAParts[i].length === 0 && partBParts[i].length > 0) {
  3539. return 1;
  3540. }
  3541. if (partAParts[i] > partBParts[i]) {
  3542. return 1;
  3543. }
  3544. if (partAParts[i] < partBParts[i]) {
  3545. return -1;
  3546. }
  3547. }
  3548. return 0;
  3549. },
  3550. // It goes number, string, number, string. If it doesn't exist, then
  3551. // 0 for numbers, empty string for strings.
  3552. parseVersionPart(part) {
  3553. if (!part) {
  3554. return [0, "", 0, ""];
  3555. }
  3556. const partParts = /([0-9]*)([^0-9]*)([0-9]*)([^0-9]*)/.exec(part);
  3557. return [
  3558. partParts[1] ? parseInt(partParts[1]) : 0,
  3559. partParts[2],
  3560. partParts[3] ? parseInt(partParts[3]) : 0,
  3561. partParts[4]
  3562. ];
  3563. },
  3564. /**
  3565. *
  3566. * @param installButton 安装按钮
  3567. * 必须属性
  3568. * + data-update-label 按钮升级的文字
  3569. * + data-downgrade-label 按钮降级的文字
  3570. * + data-reinstall-label 按钮重装的文字
  3571. * @param installedVersion 安装版本
  3572. * @param version 版本号
  3573. * @returns
  3574. */
  3575. handleInstallResult(installButton, installedVersion, version) {
  3576. if (installedVersion == null) {
  3577. return;
  3578. }
  3579. installButton.removeAttribute("data-ping-url");
  3580. switch (this.compareVersions(installedVersion, version)) {
  3581. case -1:
  3582. installButton.textContent = installButton.getAttribute("data-update-label");
  3583. break;
  3584. case 1:
  3585. installButton.textContent = installButton.getAttribute(
  3586. "data-downgrade-label"
  3587. );
  3588. break;
  3589. case 0:
  3590. installButton.textContent = installButton.getAttribute(
  3591. "data-reinstall-label"
  3592. );
  3593. break;
  3594. }
  3595. },
  3596. /**
  3597. * 检测js脚本的更新
  3598. * @param installButton 安装按钮
  3599. * 必须属性
  3600. * + data-script-name 脚本名
  3601. * + data-script-namespace 脚本命名空间
  3602. * + data-script-version 脚本当前版本号
  3603. * @param retry 重试
  3604. */
  3605. async checkForUpdatesJS(installButton, retry) {
  3606. const name = installButton.getAttribute("data-script-name");
  3607. const namespace = installButton.getAttribute("data-script-namespace");
  3608. const version = installButton.getAttribute("data-script-version");
  3609. try {
  3610. let installedVersion = await this.getInstalledVersion(name, namespace);
  3611. if (installedVersion == null) {
  3612. return false;
  3613. }
  3614. this.handleInstallResult(installButton, installedVersion, version);
  3615. return true;
  3616. } catch (error) {
  3617. if (retry) {
  3618. await utils.sleep(1e3);
  3619. try {
  3620. return await this.checkForUpdatesJS(installButton, false);
  3621. } catch (error2) {
  3622. }
  3623. }
  3624. return false;
  3625. }
  3626. },
  3627. /**
  3628. * 检测css脚本的更新
  3629. * @param installButton 安装按钮
  3630. * 必须属性
  3631. * + data-script-name 脚本名
  3632. * + data-script-namespace 脚本命名空间
  3633. */
  3634. checkForUpdatesCSS(installButton) {
  3635. const name = installButton.getAttribute("data-script-name");
  3636. const namespace = installButton.getAttribute("data-script-namespace");
  3637. postMessage(
  3638. { type: "style-version-query", name, namespace, url: location.href },
  3639. location.origin
  3640. );
  3641. }
  3642. };
  3643. const parseScriptListInfo = ($scriptList) => {
  3644. let dataset = $scriptList.dataset;
  3645. const info = {
  3646. scriptId: parseInt(dataset.scriptId),
  3647. scriptName: dataset.scriptName,
  3648. scriptAuthors: [],
  3649. scriptDailyInstalls: parseInt(dataset.scriptDailyInstalls),
  3650. scriptTotalInstalls: parseInt(dataset.scriptTotalInstalls),
  3651. scriptRatingScore: parseFloat(dataset.scriptRatingScore),
  3652. scriptCreatedDate: new Date(dataset.scriptCreatedDate),
  3653. scriptUpdatedDate: new Date(dataset.scriptUpdatedDate),
  3654. scriptType: dataset.scriptType,
  3655. scriptVersion: dataset.scriptVersion,
  3656. sensitive: dataset.sensitive === "true",
  3657. scriptLanguage: dataset.scriptLanguage,
  3658. cssAvailableAsJs: dataset.cssAvailableAsJs === "true",
  3659. codeUrl: dataset.codeUrl,
  3660. scriptDescription: dataset.scriptDescription,
  3661. scriptAuthorId: parseInt(dataset.scriptAuthorId),
  3662. scriptAuthorName: dataset.scriptAuthorName
  3663. };
  3664. let scriptAuthorsObj = utils.toJSON(dataset.scriptAuthors);
  3665. Object.keys(scriptAuthorsObj).forEach((authorId) => {
  3666. let authorName = scriptAuthorsObj[authorId];
  3667. info.scriptAuthors.push({
  3668. authorId: parseInt(authorId),
  3669. authorName
  3670. });
  3671. });
  3672. if ((info.scriptAuthorName == null || isNaN(info.scriptAuthorId)) && info.scriptAuthors.length) {
  3673. info.scriptAuthorName = info.scriptAuthors[0].authorName;
  3674. info.scriptAuthorId = info.scriptAuthors[0].authorId;
  3675. }
  3676. if (info.scriptDescription == null) {
  3677. let $description = $scriptList.querySelector(".script-description") || $scriptList.querySelector(".description");
  3678. if ($description) {
  3679. info.scriptDescription = $description.innerText || $description.textContent;
  3680. }
  3681. }
  3682. return info;
  3683. };
  3684. const GreasyforkScriptsList = {
  3685. init() {
  3686. PopsPanel.execMenuOnce("gf-scripts-filter-enable", () => {
  3687. GreasyforkScriptsFilter.init();
  3688. });
  3689. PopsPanel.execMenuOnce("beautifyCenterContent", () => {
  3690. return this.beautifyCenterContent();
  3691. });
  3692. },
  3693. /**
  3694. * 美化脚本列表
  3695. */
  3696. beautifyCenterContent() {
  3697. log.info("美化脚本列表-双列");
  3698. let result = [];
  3699. result.push(addStyle(beautifyCenterContentCSS));
  3700. const lodingClassName = "lum-lightbox-loader";
  3701. const noInstallBtnText = i18next.t("安装此脚本");
  3702. DOMUtils.ready(async () => {
  3703. let allScriptsList = GreasyforkScriptsFilter.getElementList();
  3704. allScriptsList.forEach(($scriptList) => {
  3705. if ($scriptList.querySelector(".script-list-operation")) {
  3706. return;
  3707. }
  3708. let scriptInfo = parseScriptListInfo($scriptList);
  3709. let $inlineStats = $scriptList.querySelector(
  3710. ".inline-script-stats"
  3711. );
  3712. if (!$inlineStats) {
  3713. log.error("美化脚本列表失败,未获取到.inline-script-stats");
  3714. return;
  3715. }
  3716. let code_url = scriptInfo.codeUrl;
  3717. let $ratingScoreLeft = DOMUtils.createElement("dt", {
  3718. className: "script-list-rating-score",
  3719. innerHTML: `<span>${i18next.t("评分")}</span>`
  3720. });
  3721. let $ratingScoreRight = DOMUtils.createElement(
  3722. "dd",
  3723. {
  3724. className: "script-list-rating-score",
  3725. innerHTML: `<span>${scriptInfo.scriptRatingScore}</span>`
  3726. },
  3727. {
  3728. "data-position": "right"
  3729. }
  3730. );
  3731. let $goodRatingCount = $scriptList.querySelector(
  3732. "dd.script-list-ratings .good-rating-count"
  3733. );
  3734. let $okRatingCount = $scriptList.querySelector(
  3735. "dd.script-list-ratings .ok-rating-count"
  3736. );
  3737. let $badRatingCount = $scriptList.querySelector(
  3738. "dd.script-list-ratings .bad-rating-count"
  3739. );
  3740. if ($goodRatingCount && $okRatingCount && $badRatingCount) {
  3741. let goodRatingCount = parseInt($goodRatingCount.innerText);
  3742. let okRatingCount = parseInt($okRatingCount.innerText);
  3743. let badRatingCount = parseInt($badRatingCount.innerText);
  3744. let totalRatingCount = goodRatingCount + okRatingCount + badRatingCount;
  3745. if (totalRatingCount >= 10) {
  3746. if (goodRatingCount / totalRatingCount >= 0.6) {
  3747. $ratingScoreRight.classList.add("good-rating-count");
  3748. } else {
  3749. $ratingScoreRight.classList.add("bad-rating-count");
  3750. }
  3751. } else if (totalRatingCount == 0) {
  3752. $ratingScoreRight.classList.add("good-rating-count");
  3753. } else {
  3754. if (goodRatingCount > okRatingCount + badRatingCount) {
  3755. $ratingScoreRight.classList.add("good-rating-count");
  3756. } else {
  3757. $ratingScoreRight.classList.add("bad-rating-count");
  3758. }
  3759. }
  3760. }
  3761. let $versionLeft = DOMUtils.createElement("dt", {
  3762. className: "script-list-version",
  3763. innerHTML: (
  3764. /*html*/
  3765. `<span>${i18next.t("版本")}</span>`
  3766. )
  3767. });
  3768. let $versionRight = DOMUtils.createElement(
  3769. "dd",
  3770. {
  3771. className: "script-list-version",
  3772. innerHTML: (
  3773. /*html*/
  3774. `<span>${scriptInfo.scriptVersion}</span>`
  3775. )
  3776. },
  3777. {
  3778. "data-position": "right"
  3779. }
  3780. );
  3781. let $operationLeft = DOMUtils.createElement("dt", {
  3782. className: "script-list-operation",
  3783. innerHTML: `<span>${i18next.t("操作")}</span>`
  3784. });
  3785. let $operationRight = DOMUtils.createElement(
  3786. "dd",
  3787. {
  3788. className: "script-list-operation",
  3789. innerHTML: (
  3790. /*html*/
  3791. `
  3792. <a
  3793. target="_blank"
  3794. class="install-link"
  3795. data-install-format="js"
  3796. data-script-name="${scriptInfo.scriptName}"
  3797. data-script-namespace=""
  3798. data-script-version="${scriptInfo.scriptVersion}"
  3799. data-update-label="${i18next.t("更新到 {{version}} 版本", {
  3800. version: scriptInfo.scriptVersion
  3801. })}"
  3802. data-downgrade-label="${i18next.t("降级到 {{version}} 版本", {
  3803. version: scriptInfo.scriptVersion
  3804. })}"
  3805. data-reinstall-label="${i18next.t("重新安装 {{version}} 版本", {
  3806. version: scriptInfo.scriptVersion
  3807. })}"
  3808. href="${code_url}"></a>
  3809. <button class="script-collect-btn">${i18next.t("收藏")}</button>
  3810. `
  3811. )
  3812. },
  3813. {
  3814. "data-position": "right",
  3815. style: "gap:10px;display: flex;flex-wrap: wrap;align-items: center;"
  3816. }
  3817. );
  3818. let $collect = $operationRight.querySelector(
  3819. ".script-collect-btn"
  3820. );
  3821. let $installLink = $operationRight.querySelector(".install-link");
  3822. $installLink["data-script-info"] = scriptInfo;
  3823. DOMUtils.addClass($installLink, lodingClassName);
  3824. if (scriptInfo.scriptType === "library") {
  3825. $installLink.remove();
  3826. }
  3827. DOMUtils.on($collect, "click", (event) => {
  3828. utils.preventEvent(event);
  3829. GreasyforkScriptsCollectEvent(scriptInfo.scriptId);
  3830. });
  3831. if (PopsPanel.getValue("gf-scripts-filter-enable")) {
  3832. let $filter = DOMUtils.createElement("button", {
  3833. className: "script-filter-btn",
  3834. innerHTML: i18next.t("过滤")
  3835. });
  3836. let attr_filter_key = "data-filter-key";
  3837. let attr_filter_value = "data-filter-value";
  3838. DOMUtils.on($filter, "click", (event) => {
  3839. utils.preventEvent(event);
  3840. let $dialog = __pops.alert({
  3841. title: {
  3842. text: i18next.t("选择需要过滤的选项"),
  3843. position: "center"
  3844. },
  3845. content: {
  3846. text: (
  3847. /*html*/
  3848. `
  3849. <button ${attr_filter_key}="scriptId" ${attr_filter_value}="^${scriptInfo.scriptId}$">${i18next.t("脚本id:{{text}}", {
  3850. text: scriptInfo.scriptId
  3851. })}</button>
  3852. <button ${attr_filter_key}="scriptName" ${attr_filter_value}="^${utils.parseStringToRegExpString(
  3853. scriptInfo.scriptName
  3854. )}$">${i18next.t("脚本名:{{text}}", {
  3855. text: scriptInfo.scriptName
  3856. })}</button>
  3857. `
  3858. ),
  3859. html: true
  3860. },
  3861. mask: {
  3862. enable: true,
  3863. clickEvent: {
  3864. toClose: true
  3865. }
  3866. },
  3867. width: "350px",
  3868. height: "300px",
  3869. drag: true,
  3870. dragLimit: true,
  3871. style: (
  3872. /*css*/
  3873. `
  3874. .pops-alert-content{
  3875. display: flex;
  3876. flex-direction: column;
  3877. gap: 20px;
  3878. }
  3879. .pops-alert-content button{
  3880. text-wrap: wrap;
  3881. padding: 8px;
  3882. height: auto;
  3883. text-align: left;
  3884. }
  3885. `
  3886. )
  3887. });
  3888. let $content = $dialog.$shadowRoot.querySelector(
  3889. ".pops-alert-content"
  3890. );
  3891. scriptInfo.scriptAuthors.forEach((scriptAuthorInfo) => {
  3892. let $authorIdButton = DOMUtils.createElement("button", {
  3893. innerHTML: i18next.t("作者id:{{text}}", {
  3894. text: scriptAuthorInfo.authorId
  3895. })
  3896. });
  3897. $authorIdButton.setAttribute(attr_filter_key, "scriptAuthorId");
  3898. $authorIdButton.setAttribute(
  3899. attr_filter_value,
  3900. "^" + scriptAuthorInfo.authorId + "$"
  3901. );
  3902. let $authorNameButton = DOMUtils.createElement("button", {
  3903. innerHTML: i18next.t("作者名:{{text}}", {
  3904. text: scriptAuthorInfo.authorName
  3905. })
  3906. });
  3907. $authorNameButton.setAttribute(
  3908. attr_filter_key,
  3909. "scriptAuthorName"
  3910. );
  3911. $authorNameButton.setAttribute(
  3912. attr_filter_value,
  3913. "^" + utils.parseStringToRegExpString(scriptAuthorInfo.authorName) + "$"
  3914. );
  3915. $content.appendChild($authorIdButton);
  3916. $content.appendChild($authorNameButton);
  3917. });
  3918. DOMUtils.on(
  3919. $dialog.$shadowRoot,
  3920. "click",
  3921. `button[${attr_filter_key}]`,
  3922. (event2) => {
  3923. utils.preventEvent(event2);
  3924. let $click = event2.target;
  3925. let key = $click.getAttribute(
  3926. attr_filter_key
  3927. );
  3928. let value = $click.getAttribute(attr_filter_value);
  3929. GreasyforkScriptsFilter.addValue(key, value);
  3930. $dialog.close();
  3931. GreasyforkScriptsFilter.filter();
  3932. Qmsg.success(i18next.t("添加成功"));
  3933. }
  3934. );
  3935. });
  3936. $operationRight.appendChild($filter);
  3937. }
  3938. $inlineStats.appendChild($ratingScoreLeft);
  3939. $inlineStats.appendChild($ratingScoreRight);
  3940. $inlineStats.appendChild($versionLeft);
  3941. $inlineStats.appendChild($versionRight);
  3942. $inlineStats.appendChild($operationLeft);
  3943. $inlineStats.appendChild($operationRight);
  3944. });
  3945. let $installLinkList = Array.from(
  3946. document.querySelectorAll(
  3947. ".install-link[data-install-format=js]"
  3948. )
  3949. );
  3950. let allScriptContainerStatus = GreasyforkCheckVersion.getScriptContainerStatus();
  3951. let hasScriptContainer = Object.values(allScriptContainerStatus).find(
  3952. (item) => item
  3953. );
  3954. let isRegisterScriptContainerNameList = GreasyforkCheckVersion.getRegisterScriptContainerNameList();
  3955. let isForceUseNameSpace = PopsPanel.getValue(
  3956. "beautifyCenterContent-queryNameSpace"
  3957. );
  3958. if (!hasScriptContainer) {
  3959. log.error("脚本容器未暴露external信息", window.external);
  3960. } else {
  3961. log.info(
  3962. "当前暴露的external信息:" + isRegisterScriptContainerNameList.map((it) => `【${it}】`).join("、")
  3963. );
  3964. }
  3965. for (let index = 0; index < $installLinkList.length; index++) {
  3966. const $installLink = $installLinkList[index];
  3967. let scriptInfo = Reflect.get(
  3968. $installLink,
  3969. "data-script-info"
  3970. );
  3971. if (hasScriptContainer) {
  3972. if (isForceUseNameSpace) {
  3973. let response = await httpx.get(
  3974. GreasyforkUrlUtils.getScriptInfoUrl(scriptInfo.scriptId),
  3975. {
  3976. fetch: true
  3977. }
  3978. );
  3979. if (response.status) {
  3980. let data = utils.toJSON(
  3981. response.data.responseText
  3982. );
  3983. $installLink.setAttribute(
  3984. "data-script-namespace",
  3985. data.namespace
  3986. );
  3987. }
  3988. }
  3989. GreasyforkCheckVersion.checkForUpdatesJS($installLink, true).then(
  3990. (checkResult) => {
  3991. DOMUtils.removeClass($installLink, lodingClassName);
  3992. if (!checkResult) {
  3993. DOMUtils.text($installLink, noInstallBtnText);
  3994. }
  3995. }
  3996. );
  3997. } else {
  3998. DOMUtils.removeClass($installLink, lodingClassName);
  3999. DOMUtils.text($installLink, noInstallBtnText);
  4000. }
  4001. }
  4002. });
  4003. return result;
  4004. }
  4005. };
  4006. const GreasyforkElementUtils = {
  4007. /**
  4008. * 获取当前登录用户id
  4009. */
  4010. getCurrentLoginUserId() {
  4011. let $anchor = document.querySelector(
  4012. "#nav-user-info .user-profile-link a"
  4013. );
  4014. if (!$anchor) {
  4015. return;
  4016. }
  4017. let userId = GreasyforkUrlUtils.getUserId($anchor.href);
  4018. if (userId == null) {
  4019. return;
  4020. }
  4021. return userId;
  4022. },
  4023. /**
  4024. * 注册顶部导航菜单
  4025. */
  4026. registerTopNavMenu(config) {
  4027. domUtils.ready(() => {
  4028. let $nav = $("#site-nav nav");
  4029. let $subNav = $("#site-nav .with-submenu nav");
  4030. if (!$nav) {
  4031. log.error("元素#site-nav nav不存在");
  4032. return;
  4033. }
  4034. let $menuLink = domUtils.createElement("li", {
  4035. className: config.className,
  4036. innerHTML: (
  4037. /*html*/
  4038. `
  4039. <a href="javascript:;">${config.name}</a>
  4040. `
  4041. )
  4042. });
  4043. domUtils.on($menuLink, "click", (event) => {
  4044. utils.preventEvent(event);
  4045. config.clickEvent(event);
  4046. });
  4047. if ($subNav && $subNav.children.length) {
  4048. $subNav.appendChild($menuLink);
  4049. } else {
  4050. $nav.appendChild($menuLink);
  4051. }
  4052. let $mobileMenuLink = domUtils.createElement("li", {
  4053. className: config.className,
  4054. innerHTML: (
  4055. /*html*/
  4056. `
  4057. <a href="javascript:;">${config.name}</a>
  4058. `
  4059. )
  4060. });
  4061. domUtils.on($mobileMenuLink, "click", (event) => {
  4062. utils.preventEvent(event);
  4063. config.clickEvent(event);
  4064. });
  4065. let $mobileNav = $("#mobile-nav nav");
  4066. let $multiLinkNav = $("#mobile-nav nav .multi-link-nav");
  4067. if ($multiLinkNav) {
  4068. domUtils.before($multiLinkNav, $mobileMenuLink);
  4069. } else {
  4070. if ($mobileNav) {
  4071. domUtils.append($mobileNav, $mobileMenuLink);
  4072. } else {
  4073. log.error("元素#mobile-nav nav不存在");
  4074. }
  4075. }
  4076. });
  4077. }
  4078. };
  4079. const GreasyforkUtils = {
  4080. /**
  4081. * 判断是否是当前已登录账户的主页
  4082. */
  4083. isCurrentLoginUserHome() {
  4084. let currentLoginUserId = GreasyforkElementUtils.getCurrentLoginUserId();
  4085. if (currentLoginUserId != null && GreasyforkRouter.isUsers() && window.location.pathname.includes("/" + currentLoginUserId)) {
  4086. return true;
  4087. } else {
  4088. return false;
  4089. }
  4090. }
  4091. };
  4092. const GreasyforkScriptsFilter = {
  4093. /** 存储的键 */
  4094. key: "gf-shield-rule",
  4095. init() {
  4096. log.info("脚本过滤");
  4097. let lockFunction = new utils.LockFunction(() => {
  4098. this.filter();
  4099. }, 50);
  4100. domUtils.ready(() => {
  4101. if (GreasyforkUtils.isCurrentLoginUserHome()) {
  4102. log.warn("当前在已登录的账户主页下,禁用脚本过滤");
  4103. return;
  4104. }
  4105. utils.mutationObserver(document.body, {
  4106. config: {
  4107. subtree: true,
  4108. childList: true
  4109. },
  4110. callback: () => {
  4111. lockFunction.run();
  4112. }
  4113. });
  4114. lockFunction.run();
  4115. });
  4116. },
  4117. /**
  4118. * 获取脚本列表元素
  4119. */
  4120. getElementList() {
  4121. let scriptList = [];
  4122. scriptList = scriptList.concat(
  4123. Array.from(document.querySelectorAll("ol.script-list li"))
  4124. );
  4125. return scriptList;
  4126. },
  4127. /**
  4128. * 对页面进行过滤
  4129. */
  4130. filter() {
  4131. this.getElementList().forEach(($scriptList) => {
  4132. let data = parseScriptListInfo($scriptList);
  4133. let localValueSplit = this.getValue().split("\n");
  4134. for (let index = 0; index < localValueSplit.length; index++) {
  4135. let localRule = localValueSplit[index];
  4136. let ruleSplit = localRule.split("##");
  4137. let ruleName = ruleSplit[0];
  4138. let ruleValue = ruleSplit[1];
  4139. if (ruleName === "scriptRatingScore") {
  4140. let userRatingScoreValue = parseFloat(ruleValue.slice(1));
  4141. if (ruleValue.startsWith(">")) {
  4142. if (data.scriptRatingScore > userRatingScoreValue) {
  4143. log.info("触发脚本过滤规则", [localRule, data]);
  4144. $scriptList.remove();
  4145. break;
  4146. }
  4147. } else if (ruleValue.startsWith("<")) {
  4148. if (data.scriptRatingScore < userRatingScoreValue) {
  4149. log.info("触发脚本过滤规则", [localRule, data]);
  4150. $scriptList.remove();
  4151. break;
  4152. }
  4153. }
  4154. } else if (ruleName in data || ruleName === "scriptDescription") {
  4155. if (typeof ruleValue !== "string") {
  4156. continue;
  4157. }
  4158. let ruleValueRegExp = new RegExp(ruleValue, "ig");
  4159. let scriptInfoString = String(data[ruleName]);
  4160. if (scriptInfoString.match(ruleValueRegExp)) {
  4161. log.info("触发脚本过滤规则", localRule, data);
  4162. $scriptList.remove();
  4163. break;
  4164. }
  4165. }
  4166. }
  4167. });
  4168. },
  4169. setValue(value) {
  4170. PopsPanel.setValue(this.key, value);
  4171. },
  4172. addValue(key, value) {
  4173. let localValue = this.getValue();
  4174. if (localValue.trim() !== "") {
  4175. localValue += "\n";
  4176. }
  4177. if (utils.isNull(key)) {
  4178. return;
  4179. }
  4180. key = key.toString().trim();
  4181. let rule = key + "##" + value;
  4182. localValue += rule;
  4183. this.setValue(localValue);
  4184. },
  4185. getValue() {
  4186. return PopsPanel.getValue(this.key, "");
  4187. }
  4188. };
  4189. const SettingUICommon = {
  4190. id: "greasy-fork-panel-config-account",
  4191. title: i18next.t("通用"),
  4192. forms: [
  4193. {
  4194. text: "",
  4195. type: "forms",
  4196. forms: [
  4197. {
  4198. text: i18next.t("Toast配置"),
  4199. type: "deepMenu",
  4200. forms: [
  4201. {
  4202. text: "",
  4203. type: "forms",
  4204. forms: [
  4205. UISelect(
  4206. i18next.t("Toast位置"),
  4207. "qmsg-config-position",
  4208. "bottom",
  4209. [
  4210. {
  4211. value: "topleft",
  4212. text: i18next.t("左上角")
  4213. },
  4214. {
  4215. value: "top",
  4216. text: i18next.t("顶部")
  4217. },
  4218. {
  4219. value: "topright",
  4220. text: i18next.t("右上角")
  4221. },
  4222. {
  4223. value: "left",
  4224. text: i18next.t("左边")
  4225. },
  4226. {
  4227. value: "center",
  4228. text: i18next.t("中间")
  4229. },
  4230. {
  4231. value: "right",
  4232. text: i18next.t("右边")
  4233. },
  4234. {
  4235. value: "bottomleft",
  4236. text: i18next.t("左下角")
  4237. },
  4238. {
  4239. value: "bottom",
  4240. text: i18next.t("底部")
  4241. },
  4242. {
  4243. value: "bottomright",
  4244. text: i18next.t("右下角")
  4245. }
  4246. ],
  4247. (event, isSelectValue, isSelectText) => {
  4248. log.info("设置当前Qmsg弹出位置" + isSelectText);
  4249. },
  4250. i18next.t("Toast显示在页面九宫格的位置")
  4251. ),
  4252. UISelect(
  4253. i18next.t("最多显示的数量"),
  4254. "qmsg-config-maxnums",
  4255. 3,
  4256. [
  4257. {
  4258. value: 1,
  4259. text: "1"
  4260. },
  4261. {
  4262. value: 2,
  4263. text: "2"
  4264. },
  4265. {
  4266. value: 3,
  4267. text: "3"
  4268. },
  4269. {
  4270. value: 4,
  4271. text: "4"
  4272. },
  4273. {
  4274. value: 5,
  4275. text: "5"
  4276. }
  4277. ],
  4278. void 0,
  4279. i18next.t("限制Toast显示的数量")
  4280. ),
  4281. UISwitch(
  4282. i18next.t("逆序弹出"),
  4283. "qmsg-config-showreverse",
  4284. false,
  4285. void 0,
  4286. i18next.t("修改Toast弹出的顺序")
  4287. )
  4288. ]
  4289. }
  4290. ]
  4291. },
  4292. UISelect(
  4293. i18next.t("语言"),
  4294. "setting-language",
  4295. "zh-CN",
  4296. [
  4297. {
  4298. value: "zh-CN",
  4299. text: "中文"
  4300. },
  4301. {
  4302. value: "en-US",
  4303. text: "English"
  4304. }
  4305. ],
  4306. (event, isSelectValue, isSelectText) => {
  4307. log.info("改变语言:" + isSelectText);
  4308. i18next.changeLanguage(isSelectValue);
  4309. }
  4310. )
  4311. ]
  4312. },
  4313. {
  4314. text: "",
  4315. type: "forms",
  4316. forms: [
  4317. {
  4318. text: i18next.t("账号/密码"),
  4319. type: "deepMenu",
  4320. forms: [
  4321. {
  4322. text: "",
  4323. type: "forms",
  4324. forms: [
  4325. UIInput(
  4326. i18next.t("账号"),
  4327. "user",
  4328. "",
  4329. void 0,
  4330. void 0,
  4331. i18next.t("请输入账号")
  4332. ),
  4333. UIInput(
  4334. i18next.t("密码"),
  4335. "pwd",
  4336. "",
  4337. void 0,
  4338. void 0,
  4339. i18next.t("请输入密码"),
  4340. false,
  4341. true
  4342. )
  4343. ]
  4344. },
  4345. {
  4346. text: "",
  4347. type: "forms",
  4348. forms: [
  4349. UISwitch(
  4350. i18next.t("自动登录"),
  4351. "autoLogin",
  4352. true,
  4353. void 0,
  4354. i18next.t("自动登录当前保存的账号")
  4355. ),
  4356. UIButton(
  4357. i18next.t("清空账号/密码"),
  4358. void 0,
  4359. i18next.t("点击清空"),
  4360. void 0,
  4361. void 0,
  4362. false,
  4363. "default",
  4364. (event) => {
  4365. if (confirm(i18next.t("确定清空账号和密码?"))) {
  4366. PopsPanel.deleteValue("user");
  4367. PopsPanel.deleteValue("pwd");
  4368. Qmsg.success(i18next.t("已清空账号/密码"));
  4369. let $shadowRoot = event.target.getRootNode();
  4370. $shadowRoot.querySelector(
  4371. `li[data-key="user"] .pops-panel-input input`
  4372. ).value = "";
  4373. $shadowRoot.querySelector(
  4374. `li[data-key="pwd"] .pops-panel-input input`
  4375. ).value = "";
  4376. }
  4377. }
  4378. )
  4379. ]
  4380. }
  4381. ]
  4382. },
  4383. {
  4384. text: i18next.t("功能"),
  4385. type: "deepMenu",
  4386. forms: [
  4387. {
  4388. text: i18next.t("功能"),
  4389. type: "forms",
  4390. forms: [
  4391. UISelect(
  4392. i18next.t("固定当前语言"),
  4393. "language-selector-locale",
  4394. "",
  4395. function() {
  4396. let result = [
  4397. {
  4398. value: "",
  4399. text: i18next.t("无")
  4400. }
  4401. ];
  4402. document.querySelectorAll(
  4403. "select#language-selector-locale option"
  4404. ).forEach((element) => {
  4405. let value = element.getAttribute("value");
  4406. if (value === "help") {
  4407. return;
  4408. }
  4409. let text = (element.innerText || element.textContent).trim();
  4410. result.push({
  4411. value,
  4412. text
  4413. });
  4414. });
  4415. return result;
  4416. }()
  4417. ),
  4418. UISwitch(
  4419. i18next.t("修复图片宽度显示问题"),
  4420. "fixImageWidth",
  4421. true,
  4422. void 0,
  4423. i18next.t("修复图片在移动端宽度超出浏览器宽度问题")
  4424. ),
  4425. UISwitch(
  4426. i18next.t("优化图片浏览"),
  4427. "optimizeImageBrowsing",
  4428. true,
  4429. void 0,
  4430. i18next.t("使用Viewer浏览图片")
  4431. ),
  4432. UISwitch(
  4433. i18next.t("覆盖图床图片跳转"),
  4434. "overlayBedImageClickEvent",
  4435. true,
  4436. void 0,
  4437. i18next.t("配合上面的【优化图片浏览】更优雅浏览图片")
  4438. ),
  4439. UISwitch(
  4440. i18next.t("添加【操作面板】按钮"),
  4441. "scripts-addOperationPanelBtnWithNavigator",
  4442. true,
  4443. void 0,
  4444. i18next.t("在脚本列表页面时为顶部导航栏添加【操作面板】按钮")
  4445. ),
  4446. UISwitch(
  4447. i18next.t("给Markdown添加【复制】按钮"),
  4448. "addMarkdownCopyButton",
  4449. true,
  4450. void 0,
  4451. i18next.t(
  4452. "在Markdown内容右上角添加【复制】按钮,点击一键复制Markdown内容"
  4453. )
  4454. )
  4455. ]
  4456. },
  4457. {
  4458. text: i18next.t("检测页面加载"),
  4459. type: "forms",
  4460. forms: [
  4461. UISwitch(
  4462. i18next.t("启用"),
  4463. "checkPage",
  4464. true,
  4465. void 0,
  4466. i18next.t(
  4467. "检测Greasyfork页面是否正常加载,如加载失败则自动刷新页面"
  4468. )
  4469. ),
  4470. UISelect(
  4471. i18next.t("检测间隔"),
  4472. "greasyfork-check-page-timeout",
  4473. 5,
  4474. (() => {
  4475. let result = [];
  4476. for (let index = 0; index < 5; index++) {
  4477. result.push({
  4478. value: index + 1,
  4479. text: index + 1 + "s"
  4480. });
  4481. }
  4482. return result;
  4483. })(),
  4484. void 0,
  4485. i18next.t(
  4486. "设置检测上次刷新页面的间隔时间,当距离上次刷新页面的时间超过设置的值,将不再刷新页面"
  4487. )
  4488. )
  4489. ]
  4490. }
  4491. ]
  4492. },
  4493. {
  4494. type: "deepMenu",
  4495. text: i18next.t("表单"),
  4496. forms: [
  4497. {
  4498. type: "forms",
  4499. text: "",
  4500. forms: [
  4501. UISwitch(
  4502. i18next.t("记住回复内容"),
  4503. "rememberReplyContent",
  4504. true,
  4505. void 0,
  4506. i18next.t(
  4507. "监听表单内的textarea内容改变并存储到indexDB中,提交表单将清除保存的数据,误刷新页面时可动态恢复"
  4508. )
  4509. ),
  4510. UISelect(
  4511. i18next.t("自动清理空间"),
  4512. "gf-autoClearRememberReplayContent",
  4513. 7,
  4514. [
  4515. {
  4516. text: i18next.t("不清理"),
  4517. value: -1
  4518. },
  4519. {
  4520. text: i18next.t("{{value}} 天", {
  4521. value: 1
  4522. }),
  4523. value: 1
  4524. },
  4525. {
  4526. text: i18next.t("{{value}} 周", {
  4527. value: 1
  4528. }),
  4529. value: 7
  4530. },
  4531. {
  4532. text: i18next.t("{{value}} 个月", {
  4533. value: 1
  4534. }),
  4535. value: 30
  4536. },
  4537. {
  4538. text: i18next.t("{{value}} 个月", {
  4539. value: 2
  4540. }),
  4541. value: 60
  4542. },
  4543. {
  4544. text: i18next.t("{{value}} 个月", {
  4545. value: 3
  4546. }),
  4547. value: 90
  4548. },
  4549. {
  4550. text: i18next.t("半年"),
  4551. value: 180
  4552. }
  4553. ],
  4554. void 0,
  4555. i18next.t("根据设置的间隔时间自动清理保存的回复内容")
  4556. ),
  4557. UIButton(
  4558. i18next.t(`数据占用空间:{{size}}`, {
  4559. size: i18next.t("计算中")
  4560. }),
  4561. i18next.t("当前存储的数据所占用的空间大小"),
  4562. i18next.t("清空"),
  4563. void 0,
  4564. void 0,
  4565. void 0,
  4566. "default",
  4567. async () => {
  4568. let isClear = await GreasyforkRememberFormTextArea.clearAllRememberReplyContent();
  4569. if (isClear) {
  4570. Qmsg.success(i18next.t("清理成功"));
  4571. } else {
  4572. Qmsg.error(i18next.t("清理失败"));
  4573. }
  4574. },
  4575. async (formConfig, container) => {
  4576. let $leftTopText = container.ulElement.querySelector(
  4577. 'li[data-key="gf-autoClearRememberReplayContent"]+li .pops-panel-item-left-main-text'
  4578. );
  4579. let allText = await GreasyforkRememberFormTextArea.getAllRememberReplyContent();
  4580. let showSize = "";
  4581. if (allText.length) {
  4582. showSize = utils.getTextStorageSize(
  4583. JSON.stringify(allText)
  4584. );
  4585. } else {
  4586. showSize = utils.getTextStorageSize("");
  4587. }
  4588. $leftTopText.innerText = i18next.t(
  4589. `数据占用空间:{{size}}`,
  4590. {
  4591. size: showSize
  4592. }
  4593. );
  4594. }
  4595. )
  4596. ]
  4597. }
  4598. ]
  4599. },
  4600. {
  4601. text: i18next.t("美化"),
  4602. type: "deepMenu",
  4603. forms: [
  4604. {
  4605. text: i18next.t("全局"),
  4606. type: "forms",
  4607. forms: [
  4608. UISwitch(
  4609. i18next.t("美化页面元素"),
  4610. "beautifyPage",
  4611. true,
  4612. void 0,
  4613. i18next.t("如button、input、textarea")
  4614. ),
  4615. UISwitch(
  4616. i18next.t("美化上传图片按钮"),
  4617. "beautifyUploadImage",
  4618. true,
  4619. void 0,
  4620. i18next.t("放大上传区域")
  4621. ),
  4622. UISwitch(
  4623. i18next.t("美化顶部导航栏"),
  4624. "beautifyTopNavigationBar",
  4625. true,
  4626. void 0,
  4627. i18next.t("可能会跟Greasyfork Beautify脚本有冲突")
  4628. ),
  4629. UISwitch(
  4630. i18next.t("美化Greasyfork Beautify脚本"),
  4631. "beautifyGreasyforkBeautify",
  4632. true,
  4633. void 0,
  4634. i18next.t(
  4635. '需安装Greasyfork Beautify脚本,<a href="https://greasyfork.org/zh-CN/scripts/446849-greasyfork-beautify" target="_blank">🖐点我安装</a>'
  4636. )
  4637. )
  4638. ]
  4639. },
  4640. {
  4641. type: "forms",
  4642. text: i18next.t("脚本列表"),
  4643. forms: [
  4644. UISwitch(
  4645. i18next.t("美化脚本列表"),
  4646. "beautifyCenterContent",
  4647. true,
  4648. void 0,
  4649. i18next.t("双列显示且添加脚本卡片操作项(安装、收藏)")
  4650. ),
  4651. UISwitch(
  4652. "↑" + i18next.t("使用namespace查询脚本信息"),
  4653. "beautifyCenterContent-queryNameSpace",
  4654. true,
  4655. void 0,
  4656. i18next.t("开启后检测已安装的脚本信息更准确,但是速度会更慢")
  4657. )
  4658. ]
  4659. }
  4660. ]
  4661. },
  4662. {
  4663. type: "deepMenu",
  4664. text: i18next.t("自定义快捷键"),
  4665. forms: [
  4666. {
  4667. type: "forms",
  4668. text: "",
  4669. forms: [
  4670. UIButtonShortCut(
  4671. i18next.t("快捷键发表回复"),
  4672. i18next.t("在输入框内按下快捷发表回复,例如:{{key}}", {
  4673. key: "Ctrl + Enter"
  4674. }),
  4675. "gf-quickReply",
  4676. {
  4677. keyName: "Enter",
  4678. keyValue: "13",
  4679. ohterCodeList: ["ctrl"]
  4680. },
  4681. i18next.t("点击录入快捷键"),
  4682. void 0,
  4683. GreasyforkShortCut.shortCut
  4684. )
  4685. ]
  4686. }
  4687. ]
  4688. },
  4689. {
  4690. text: i18next.t("过滤"),
  4691. type: "deepMenu",
  4692. forms: [
  4693. {
  4694. 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(
  4695. "帮助文档"
  4696. )}</a>`,
  4697. type: "forms",
  4698. forms: [
  4699. UISwitch(
  4700. i18next.t("启用"),
  4701. "gf-scripts-filter-enable",
  4702. true,
  4703. void 0,
  4704. i18next.t("作用域:脚本、脚本搜索、用户主页")
  4705. ),
  4706. {
  4707. type: "own",
  4708. getLiElementCallBack(liElement) {
  4709. let textareaDiv = domUtils.createElement(
  4710. "div",
  4711. {
  4712. className: "pops-panel-textarea",
  4713. innerHTML: `
  4714. <textarea placeholder="${i18next.t(
  4715. "请输入规则,每行一个"
  4716. )}" style="height:200px;"></textarea>`
  4717. },
  4718. {
  4719. style: "width: 100%;"
  4720. }
  4721. );
  4722. let $textarea = textareaDiv.querySelector(
  4723. "textarea"
  4724. );
  4725. $textarea.value = GreasyforkScriptsFilter.getValue();
  4726. domUtils.on(
  4727. $textarea,
  4728. ["input", "propertychange"],
  4729. void 0,
  4730. utils.debounce(function(event) {
  4731. GreasyforkScriptsFilter.setValue($textarea.value);
  4732. }, 200)
  4733. );
  4734. liElement.appendChild(textareaDiv);
  4735. return liElement;
  4736. }
  4737. }
  4738. ]
  4739. }
  4740. ]
  4741. }
  4742. ]
  4743. },
  4744. {
  4745. type: "forms",
  4746. text: i18next.t("脚本管理"),
  4747. forms: [
  4748. {
  4749. type: "deepMenu",
  4750. text: i18next.t("代码同步"),
  4751. forms: [
  4752. {
  4753. text: i18next.t("代码同步"),
  4754. type: "forms",
  4755. forms: [
  4756. UIButton(
  4757. i18next.t("源代码同步【脚本列表】"),
  4758. void 0,
  4759. i18next.t("一键同步"),
  4760. void 0,
  4761. void 0,
  4762. false,
  4763. "primary",
  4764. (event) => {
  4765. if (!GreasyforkRouter.isUsers()) {
  4766. PopsPanel.setValue(
  4767. "goto_updateSettingsAndSynchronize_scriptList",
  4768. true
  4769. );
  4770. if (GreasyforkMenu.getUserLinkElement()) {
  4771. Qmsg.success(i18next.t("前往用户主页"));
  4772. window.location.href = GreasyforkMenu.getUserLinkElement().href;
  4773. } else {
  4774. Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
  4775. }
  4776. return;
  4777. }
  4778. let scriptUrlList = [];
  4779. document.querySelectorAll(
  4780. "#user-script-list-section li a.script-link"
  4781. ).forEach((item) => {
  4782. scriptUrlList = scriptUrlList.concat(
  4783. GreasyforkUrlUtils.getAdminUrl(item.href)
  4784. );
  4785. });
  4786. GreasyforkMenu.updateScript(scriptUrlList);
  4787. }
  4788. ),
  4789. UIButton(
  4790. i18next.t("源代码同步【未上架的脚本】"),
  4791. void 0,
  4792. i18next.t("一键同步"),
  4793. void 0,
  4794. void 0,
  4795. false,
  4796. "primary",
  4797. (event) => {
  4798. if (!GreasyforkRouter.isUsers()) {
  4799. PopsPanel.setValue(
  4800. "goto_updateSettingsAndSynchronize_unlistedScriptList",
  4801. true
  4802. );
  4803. if (GreasyforkMenu.getUserLinkElement()) {
  4804. Qmsg.success(i18next.t("前往用户主页"));
  4805. window.location.href = GreasyforkMenu.getUserLinkElement().href;
  4806. } else {
  4807. Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
  4808. }
  4809. return;
  4810. }
  4811. let scriptUrlList = [];
  4812. document.querySelectorAll(
  4813. "#user-unlisted-script-list li a.script-link"
  4814. ).forEach((item) => {
  4815. scriptUrlList = scriptUrlList.concat(
  4816. GreasyforkUrlUtils.getAdminUrl(item.href)
  4817. );
  4818. });
  4819. GreasyforkMenu.updateScript(scriptUrlList);
  4820. }
  4821. ),
  4822. UIButton(
  4823. i18next.t("源代码同步【库】"),
  4824. void 0,
  4825. i18next.t("一键同步"),
  4826. void 0,
  4827. void 0,
  4828. false,
  4829. "primary",
  4830. (event) => {
  4831. if (!GreasyforkRouter.isUsers()) {
  4832. PopsPanel.setValue(
  4833. "goto_updateSettingsAndSynchronize_libraryScriptList",
  4834. true
  4835. );
  4836. if (GreasyforkMenu.getUserLinkElement()) {
  4837. Qmsg.success(i18next.t("前往用户主页"));
  4838. window.location.href = GreasyforkMenu.getUserLinkElement().href;
  4839. } else {
  4840. Qmsg.error(i18next.t("获取当前已登录的用户主页失败"));
  4841. }
  4842. return;
  4843. }
  4844. let scriptUrlList = [];
  4845. document.querySelectorAll(
  4846. "#user-library-script-list li a.script-link"
  4847. ).forEach((item) => {
  4848. scriptUrlList = scriptUrlList.concat(
  4849. GreasyforkUrlUtils.getAdminUrl(item.href)
  4850. );
  4851. });
  4852. GreasyforkMenu.updateScript(scriptUrlList);
  4853. }
  4854. )
  4855. ]
  4856. }
  4857. ]
  4858. },
  4859. {
  4860. type: "deepMenu",
  4861. text: i18next.t("脚本列表"),
  4862. forms: [],
  4863. afterEnterDeepMenuCallBack(formConfig, container) {
  4864. PopsPanelUISetting.UIScriptList(
  4865. "script-list",
  4866. container.sectionBodyContainer
  4867. );
  4868. }
  4869. },
  4870. {
  4871. type: "deepMenu",
  4872. text: i18next.t("库"),
  4873. forms: [],
  4874. afterEnterDeepMenuCallBack(formConfig, container) {
  4875. PopsPanelUISetting.UIScriptList(
  4876. "script-library",
  4877. container.sectionBodyContainer
  4878. );
  4879. }
  4880. }
  4881. ]
  4882. }
  4883. ]
  4884. };
  4885. const SettingUIScripts = {
  4886. id: "greasy-fork-panel-config-scripts",
  4887. title: i18next.t("脚本"),
  4888. forms: [
  4889. {
  4890. text: "",
  4891. type: "forms",
  4892. forms: [
  4893. {
  4894. text: i18next.t("代码"),
  4895. type: "deepMenu",
  4896. forms: [
  4897. {
  4898. text: "",
  4899. type: "forms",
  4900. forms: [
  4901. UISwitch(
  4902. i18next.t("添加复制代码按钮"),
  4903. "addCopyCodeButton",
  4904. true,
  4905. void 0,
  4906. i18next.t("更优雅的复制")
  4907. ),
  4908. UISwitch(
  4909. i18next.t("快捷键"),
  4910. "fullScreenOptimization",
  4911. true,
  4912. void 0,
  4913. i18next.t("【F】键全屏、【Alt+Shift+F】键宽屏")
  4914. ),
  4915. UISwitch(
  4916. i18next.t("修复代码行号显示"),
  4917. "code-repairCodeLineNumber",
  4918. false,
  4919. void 0,
  4920. i18next.t("修复代码行数超过1k行号显示不全问题")
  4921. ),
  4922. UISwitch(
  4923. "monacoEditor",
  4924. "code-use-monaco-editor",
  4925. true,
  4926. void 0,
  4927. i18next.t("使用Monaco编辑器")
  4928. )
  4929. ]
  4930. }
  4931. ]
  4932. },
  4933. {
  4934. text: i18next.t("历史版本"),
  4935. type: "deepMenu",
  4936. forms: [
  4937. {
  4938. text: i18next.t("功能"),
  4939. type: "forms",
  4940. forms: [
  4941. UISwitch(
  4942. i18next.t("添加额外的标签按钮"),
  4943. "scripts-versions-addExtraTagButton",
  4944. true,
  4945. void 0,
  4946. i18next.t("在版本下面添加【安装】、【查看代码】按钮")
  4947. ),
  4948. UISwitch(
  4949. i18next.t("添加代码对比按钮"),
  4950. "scripts-versions-addCompareCodeButton",
  4951. true,
  4952. void 0,
  4953. i18next.t("monacoEditor")
  4954. )
  4955. ]
  4956. },
  4957. {
  4958. text: i18next.t("美化"),
  4959. type: "forms",
  4960. forms: [
  4961. UISwitch(
  4962. i18next.t("美化历史版本页面"),
  4963. "beautifyHistoryVersionPage",
  4964. true,
  4965. void 0,
  4966. i18next.t("更直观的查看版本迭代")
  4967. )
  4968. ]
  4969. }
  4970. ]
  4971. }
  4972. ]
  4973. },
  4974. {
  4975. text: "",
  4976. type: "forms",
  4977. forms: [
  4978. {
  4979. text: i18next.t("功能"),
  4980. type: "deepMenu",
  4981. forms: [
  4982. {
  4983. text: "",
  4984. type: "forms",
  4985. forms: [
  4986. UISwitch(
  4987. i18next.t("添加【寻找引用】按钮"),
  4988. "addFindReferenceButton",
  4989. true,
  4990. void 0,
  4991. i18next.t("在脚本栏添加按钮,一般用于搜索引用该库的相关脚本")
  4992. ),
  4993. UISwitch(
  4994. i18next.t("添加【收藏】按钮"),
  4995. "addCollectionButton",
  4996. true,
  4997. void 0,
  4998. i18next.t("在脚本栏添加按钮,一般用于快捷收藏该脚本/库")
  4999. ),
  5000. UISwitch(
  5001. i18next.t("添加【今日检查】信息块"),
  5002. "scriptHomepageAddedTodaySUpdate",
  5003. true,
  5004. void 0,
  5005. i18next.t("在脚本信息栏添加【今日检查】信息块")
  5006. )
  5007. ]
  5008. }
  5009. ]
  5010. }
  5011. ]
  5012. }
  5013. ]
  5014. };
  5015. const GreasyforkDiscussionsFilter = {
  5016. /** 存储的键 */
  5017. key: "gf-discuessions-filter-rule",
  5018. $data: {
  5019. /** 脚本 */
  5020. FILTER_SCRIPT_KEY: "greasyfork-discussions-filter-script",
  5021. /** 发布用户 */
  5022. FILTER_POST_USER_KEY: "greasyfork-discussions-filter-post-user",
  5023. /** 回复用户 */
  5024. FILTER_REPLY_USER_KEY: "greasyfork-discussions-filter-reply-user"
  5025. },
  5026. init() {
  5027. log.info("论坛-过滤");
  5028. addStyle(
  5029. /*css*/
  5030. `
  5031. .discussion-list-container {
  5032. --discusstion-repeat-color: #ffa700;
  5033. }
  5034. .discussion-list-container a.discussion-title[data-repeat-tip-show]::before {
  5035. content: attr(data-repeat-tip-show);
  5036. color: var(--discusstion-repeat-color);
  5037. border-radius: 5px;
  5038. border: 2px solid var(--discusstion-repeat-color);
  5039. padding: 2px 5px;
  5040. font-weight: 800;
  5041. font-size: 14px;
  5042. }
  5043. `
  5044. );
  5045. let lockFunction = new utils.LockFunction(() => {
  5046. this.filter();
  5047. }, 50);
  5048. utils.mutationObserver(document.body, {
  5049. config: {
  5050. subtree: true,
  5051. childList: true
  5052. },
  5053. callback: () => {
  5054. lockFunction.run();
  5055. }
  5056. });
  5057. lockFunction.run();
  5058. },
  5059. /**
  5060. * 获取反馈列表元素
  5061. */
  5062. getElementList() {
  5063. let discussionsListContainer = [];
  5064. discussionsListContainer = discussionsListContainer.concat(
  5065. Array.from(
  5066. document.querySelectorAll(".discussion-list-container")
  5067. )
  5068. );
  5069. return discussionsListContainer;
  5070. },
  5071. /**
  5072. * 论坛-过滤
  5073. */
  5074. filter() {
  5075. this.transformOldRule();
  5076. const SNIPPET_MAP = /* @__PURE__ */ new Map();
  5077. this.getElementList().forEach(($listContainer, index) => {
  5078. const discussionInfo = this.parseDiscuessionListContainerInfo($listContainer);
  5079. let localValueSplit = this.getValue().split("\n");
  5080. if (SNIPPET_MAP.has(discussionInfo.snippet) && PopsPanel.getValue("greasyfork-discussions-filter-duplicate-comments")) {
  5081. let discussionTitleElement = SNIPPET_MAP.get(
  5082. discussionInfo.snippet
  5083. ).querySelector("a.discussion-title");
  5084. discussionTitleElement.setAttribute("data-repeat-tip-show", "true");
  5085. let oldCount = 0;
  5086. if (discussionTitleElement.hasAttribute("data-repeat-count")) {
  5087. oldCount = parseInt(
  5088. discussionTitleElement.getAttribute("data-repeat-count")
  5089. );
  5090. }
  5091. oldCount++;
  5092. discussionTitleElement.setAttribute(
  5093. "data-repeat-count",
  5094. oldCount.toString()
  5095. );
  5096. discussionTitleElement.setAttribute(
  5097. "data-repeat-tip-show",
  5098. i18next.t("已过滤:{{oldCount}}", { oldCount })
  5099. );
  5100. log.success([
  5101. `过滤重复内容:${discussionInfo.snippet}`,
  5102. discussionInfo
  5103. ]);
  5104. $listContainer.remove();
  5105. return;
  5106. }
  5107. SNIPPET_MAP.set(discussionInfo.snippet, $listContainer);
  5108. for (let index2 = 0; index2 < localValueSplit.length; index2++) {
  5109. let localRule = localValueSplit[index2];
  5110. let ruleSplit = localRule.split("##");
  5111. let ruleName = ruleSplit[0];
  5112. let ruleValue = ruleSplit[1];
  5113. if (ruleName in discussionInfo) {
  5114. let ruleValueRegExp = new RegExp(ruleValue, "ig");
  5115. if (discussionInfo[ruleName] != null) {
  5116. let scriptInfoString = String(discussionInfo[ruleName]);
  5117. if (scriptInfoString.match(ruleValueRegExp)) {
  5118. log.info("触发论坛过滤规则", localRule, discussionInfo);
  5119. $listContainer.remove();
  5120. return;
  5121. }
  5122. }
  5123. }
  5124. }
  5125. });
  5126. },
  5127. /**
  5128. * 解析出元素上的属性
  5129. */
  5130. parseDiscuessionListContainerInfo($listContainer) {
  5131. var _a2, _b, _c, _d;
  5132. let discussionUrl = $listContainer.querySelector("a.discussion-title").href;
  5133. let discuessionIdMatch = discussionUrl.match(
  5134. /\/discussions(|\/greasyfork)\/([\d]+)/
  5135. );
  5136. let discuessionId = discuessionIdMatch[discuessionIdMatch.length - 1];
  5137. const info = {
  5138. /** 脚本名 */
  5139. scriptName: $listContainer.querySelector(
  5140. ".discussion-meta-item-script-name"
  5141. ).innerText,
  5142. /** 脚本主页地址 */
  5143. scriptUrl: (_a2 = $listContainer.querySelector(
  5144. ".discussion-meta-item-script-name a"
  5145. )) == null ? void 0 : _a2.href,
  5146. /** 脚本id */
  5147. scriptId: GreasyforkUrlUtils.getScriptId(
  5148. (_b = $listContainer.querySelector(
  5149. ".discussion-meta-item-script-name a"
  5150. )) == null ? void 0 : _b.href
  5151. ),
  5152. /** 发布的用户名 */
  5153. postUserName: $listContainer.querySelector("a.user-link").innerText,
  5154. /** 发布的用户主页地址 */
  5155. postUserHomeUrl: $listContainer.querySelector("a.user-link").href,
  5156. /** 发布的用户id */
  5157. postUserId: GreasyforkUrlUtils.getUserId(
  5158. $listContainer.querySelector("a.user-link").href
  5159. ),
  5160. /** 发布的时间 */
  5161. postTimeStamp: new Date(
  5162. $listContainer.querySelector("relative-time").getAttribute("datetime")
  5163. ),
  5164. /** 发布的id */
  5165. snippetId: discuessionId,
  5166. /** 发布的地址*/
  5167. snippetUrl: discussionUrl,
  5168. /** 发布的内容片段*/
  5169. snippet: ((_c = $listContainer.querySelector("span.discussion-snippet")) == null ? void 0 : _c.innerText) || "",
  5170. /** (如果有)回复的用户名*/
  5171. replyUserName: void 0,
  5172. /** (如果有)回复的用户主页地址*/
  5173. replyUserHomeUrl: void 0,
  5174. /** (如果有)回复的用户id*/
  5175. replyUserId: void 0,
  5176. /** (如果有)回复的时间 */
  5177. replyTimeStamp: void 0
  5178. };
  5179. if ($listContainer.querySelector(
  5180. ".discussion-meta-item .discussion-meta-item"
  5181. )) {
  5182. info.replyUserName = $listContainer.querySelector(
  5183. ".discussion-meta-item .discussion-meta-item a.user-link"
  5184. ).innerText;
  5185. info.replyUserHomeUrl = $listContainer.querySelector(
  5186. ".discussion-meta-item .discussion-meta-item a.user-link"
  5187. ).href;
  5188. info.replyUserId = GreasyforkUrlUtils.getUserId(info.replyUserHomeUrl);
  5189. info.replyTimeStamp = new Date(
  5190. (_d = $listContainer.querySelector(
  5191. ".discussion-meta-item .discussion-meta-item relative-time"
  5192. )) == null ? void 0 : _d.getAttribute("datetime")
  5193. );
  5194. }
  5195. return info;
  5196. },
  5197. /** 转换旧规则 @deprecated */
  5198. transformOldRule() {
  5199. if (Date.now() > (/* @__PURE__ */ new Date("2024-8-19")).getTime()) {
  5200. return;
  5201. }
  5202. const FILTER_SCRIPT_KEY = "greasyfork-discussions-filter-script";
  5203. const FILTER_POST_USER_KEY = "greasyfork-discussions-filter-post-user";
  5204. const FILTER_REPLY_USER_KEY = "greasyfork-discussions-filter-reply-user";
  5205. const filterScript = PopsPanel.getValue(FILTER_SCRIPT_KEY, "");
  5206. const filterPostUser = PopsPanel.getValue(FILTER_POST_USER_KEY, "");
  5207. const filterReplyUser = PopsPanel.getValue(FILTER_REPLY_USER_KEY, "");
  5208. const filterScriptList = filterScript.trim() === "" ? [] : filterScript.split("\n");
  5209. const filterPostUserList = filterPostUser.trim() === "" ? [] : filterPostUser.split("\n");
  5210. const filterReplyUserList = filterReplyUser.trim() === "" ? [] : filterReplyUser.split("\n");
  5211. filterScriptList.forEach((ruleValue) => {
  5212. this.addValue(
  5213. "scriptId",
  5214. utils.parseStringToRegExpString("^" + ruleValue + "$")
  5215. );
  5216. });
  5217. filterPostUserList.forEach((ruleValue) => {
  5218. this.addValue(
  5219. "postUserId",
  5220. utils.parseStringToRegExpString("^" + ruleValue + "$")
  5221. );
  5222. });
  5223. filterReplyUserList.forEach((ruleValue) => {
  5224. this.addValue(
  5225. "replyUserId",
  5226. utils.parseStringToRegExpString("^" + ruleValue + "$")
  5227. );
  5228. });
  5229. PopsPanel.deleteValue(FILTER_SCRIPT_KEY);
  5230. PopsPanel.deleteValue(FILTER_POST_USER_KEY);
  5231. PopsPanel.deleteValue(FILTER_REPLY_USER_KEY);
  5232. },
  5233. setValue(value) {
  5234. PopsPanel.setValue(this.key, value);
  5235. },
  5236. addValue(key, value) {
  5237. let localValue = this.getValue();
  5238. if (localValue.trim() !== "") {
  5239. localValue += "\n";
  5240. }
  5241. if (utils.isNull(key)) {
  5242. return;
  5243. }
  5244. key = key.toString().trim();
  5245. let rule = key + "##" + value;
  5246. localValue += rule;
  5247. this.setValue(localValue);
  5248. },
  5249. getValue() {
  5250. return PopsPanel.getValue(this.key, "");
  5251. }
  5252. };
  5253. const SettingUIDiscuessions = {
  5254. id: "greasy-fork-panel-config-discussions",
  5255. title: i18next.t("论坛"),
  5256. forms: [
  5257. {
  5258. text: "",
  5259. type: "forms",
  5260. forms: [
  5261. {
  5262. text: i18next.t("功能"),
  5263. type: "deepMenu",
  5264. forms: [
  5265. {
  5266. text: "",
  5267. type: "forms",
  5268. forms: [
  5269. {
  5270. type: "own",
  5271. attributes: {
  5272. "data-key": "discussions-readBgColor",
  5273. "data-default-value": "#e5e5e5"
  5274. },
  5275. getLiElementCallBack(liElement) {
  5276. let key = "discussions-readBgColor";
  5277. let $left = domUtils.createElement("div", {
  5278. className: "pops-panel-item-left-text",
  5279. innerHTML: `
  5280. <p class="pops-panel-item-left-main-text">${i18next.t("自定义已读颜色")}</p>
  5281. <p class="pops-panel-item-left-desc-text">${i18next.t("在讨论内生效")}</p>
  5282. `
  5283. });
  5284. let $right = domUtils.createElement("div", {
  5285. className: "pops-panel-item-right",
  5286. innerHTML: `
  5287. <input type="color" class="pops-color-choose" />
  5288. `
  5289. });
  5290. let $color = $right.querySelector(
  5291. ".pops-color-choose"
  5292. );
  5293. $color.value = PopsPanel.getValue(key);
  5294. let $style = domUtils.createElement("style");
  5295. domUtils.append(document.head, $style);
  5296. domUtils.on(
  5297. $color,
  5298. ["input", "propertychange"],
  5299. (event) => {
  5300. log.info("选择颜色:" + $color.value);
  5301. $style.innerHTML = `
  5302. .discussion-read{
  5303. background: ${$color.value} !important;
  5304. }
  5305. `;
  5306. PopsPanel.setValue(key, $color.value);
  5307. }
  5308. );
  5309. liElement.appendChild($left);
  5310. liElement.appendChild($right);
  5311. return liElement;
  5312. }
  5313. },
  5314. UISwitch(
  5315. i18next.t("添加【过滤】按钮"),
  5316. "discussions-addShortcutOperationButton",
  5317. true,
  5318. void 0,
  5319. i18next.t(
  5320. "在每一行讨论的最后面添加【过滤】按钮,需开启过滤功能才会生效"
  5321. )
  5322. ),
  5323. UISwitch(
  5324. i18next.t("添加【举报】按钮"),
  5325. "discussions-addReportButton",
  5326. true,
  5327. void 0,
  5328. i18next.t("在每一行讨论的最后面添加【举报】按钮")
  5329. )
  5330. ]
  5331. }
  5332. ]
  5333. },
  5334. {
  5335. text: i18next.t("过滤"),
  5336. type: "deepMenu",
  5337. forms: [
  5338. {
  5339. 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(
  5340. "帮助文档"
  5341. )}</a>`,
  5342. type: "forms",
  5343. forms: [
  5344. UISwitch(
  5345. i18next.t("启用"),
  5346. "greasyfork-discussions-filter-enable",
  5347. true,
  5348. void 0,
  5349. i18next.t("开启后下面的过滤功能才会生效")
  5350. ),
  5351. UISwitch(
  5352. i18next.t("过滤重复的评论"),
  5353. "greasyfork-discussions-filter-duplicate-comments",
  5354. false,
  5355. void 0,
  5356. i18next.t("过滤掉重复的评论数量(≥2)")
  5357. ),
  5358. {
  5359. type: "own",
  5360. getLiElementCallBack(liElement) {
  5361. let textareaDiv = domUtils.createElement(
  5362. "div",
  5363. {
  5364. className: "pops-panel-textarea",
  5365. innerHTML: `
  5366. <textarea placeholder="${i18next.t(
  5367. "请输入规则,每行一个"
  5368. )}" style="height:200px;"></textarea>`
  5369. },
  5370. {
  5371. style: "width: 100%;"
  5372. }
  5373. );
  5374. let $textarea = textareaDiv.querySelector(
  5375. "textarea"
  5376. );
  5377. $textarea.value = GreasyforkDiscussionsFilter.getValue();
  5378. domUtils.on(
  5379. $textarea,
  5380. ["input", "propertychange"],
  5381. void 0,
  5382. utils.debounce(function(event) {
  5383. GreasyforkDiscussionsFilter.setValue($textarea.value);
  5384. }, 200)
  5385. );
  5386. liElement.appendChild(textareaDiv);
  5387. return liElement;
  5388. }
  5389. }
  5390. ]
  5391. }
  5392. ]
  5393. }
  5394. ]
  5395. }
  5396. ]
  5397. };
  5398. const UIScriptListCSS = '.w-script-list-item {\r\n padding: 10px;\r\n border-bottom: 1px solid #e5e5e5;\r\n font-size: 16px;\r\n text-align: left;\r\n background: var(--aside-bg-color);\r\n border-radius: 8px;\r\n --pops-panel-forms-margin-left-right: 10px;\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';
  5399. const SettingUIUsers = {
  5400. id: "greasy-fork-panel-config-account",
  5401. title: i18next.t("用户"),
  5402. forms: [
  5403. {
  5404. text: "",
  5405. type: "forms",
  5406. forms: [
  5407. {
  5408. text: i18next.t("功能"),
  5409. type: "deepMenu",
  5410. forms: [
  5411. {
  5412. text: "",
  5413. type: "forms",
  5414. forms: [
  5415. UISwitch(
  5416. i18next.t("迁移【控制台】到顶部导航栏"),
  5417. "users-changeConsoleToTopNavigator",
  5418. true,
  5419. void 0,
  5420. i18next.t("将【控制台】按钮移动到顶部导航栏,节省空间")
  5421. )
  5422. ]
  5423. }
  5424. ]
  5425. },
  5426. {
  5427. text: i18next.t("美化"),
  5428. type: "deepMenu",
  5429. forms: [
  5430. {
  5431. text: "",
  5432. type: "forms",
  5433. forms: [
  5434. UISwitch(
  5435. i18next.t("美化私信页面"),
  5436. "conversations-beautifyDialogBox",
  5437. true,
  5438. void 0,
  5439. i18next.t("美化为左右对话模式")
  5440. ),
  5441. UISwitch(
  5442. i18next.t("美化私信列表"),
  5443. "conversations-beautifyPrivateMessageList",
  5444. true
  5445. )
  5446. ]
  5447. }
  5448. ]
  5449. }
  5450. ]
  5451. }
  5452. ]
  5453. };
  5454. const GithubUrl2WebhookUrl = {
  5455. init() {
  5456. },
  5457. /**
  5458. * 显示视图
  5459. */
  5460. showView() {
  5461. let $alert = __pops.alert({
  5462. title: {
  5463. text: i18next.t("Url To WebhookUrl"),
  5464. position: "center"
  5465. },
  5466. content: {
  5467. text: (
  5468. /*html*/
  5469. `
  5470. <div class="github-2-webhook-container">
  5471. <div class="url-container">
  5472. <h4>Github Url</h4>
  5473. <div class="url-parse url-parse-link">
  5474. <label>${i18next.t("转换前")}</label>
  5475. <textarea id="github" placeholder="${i18next.t("例如:") + "https://github.com/WhiteSevs/TamperMonkeyScript/blob/master/README.md"}"></textarea>
  5476. </div>
  5477. <div class="url-parse url-parse-result">
  5478. <label>${i18next.t("转换后")}</label>
  5479. <textarea id="webhook" placeholder="${i18next.t("结果:") + "https://raw.githubusercontent.com/WhiteSevs/TamperMonkeyScript/master/README.md"}" readonly></textarea>
  5480. </div>
  5481. </div>
  5482. </div>
  5483. `
  5484. ),
  5485. html: true
  5486. },
  5487. btn: {
  5488. ok: {
  5489. type: "default",
  5490. text: i18next.t("关闭")
  5491. }
  5492. },
  5493. mask: {
  5494. enable: true,
  5495. clickEvent: {
  5496. toClose: false,
  5497. toHide: false
  5498. }
  5499. },
  5500. drag: true,
  5501. width: window.innerWidth > 500 ? "500px" : "88vw",
  5502. height: window.innerHeight > 500 ? "500px" : "80vh",
  5503. style: (
  5504. /*css*/
  5505. `
  5506. .github-2-webhook-container{
  5507. display: flex;
  5508. flex-direction: column;
  5509. height: 100%;
  5510. }
  5511. .url-container{
  5512. display: flex;
  5513. flex-direction: column;
  5514. gap: 10px;
  5515. padding: 20px;
  5516. flex: 1;
  5517. }
  5518. .url-parse{
  5519. display: flex;
  5520. flex-direction: column;
  5521. flex: 1;
  5522. }
  5523. .url-container textarea{
  5524. height: 100%;
  5525. width: 100%;
  5526. position: relative;
  5527. display: block;
  5528. resize: none;
  5529. padding: 5px 11px;
  5530. box-sizing: border-box;
  5531. font-size: inherit;
  5532. font-family: inherit;
  5533. background-color: rgb(255, 255, 255, var(--pops-bg-opacity));
  5534. background-image: none;
  5535. -webkit-appearance: none;
  5536. appearance: none;
  5537. box-shadow: 0 0 0 1px #dcdfe6 inset;
  5538. border-radius: 0;
  5539. transition: box-shadow 0.2s cubic-bezier(0.645, 0.045, 0.355, 1);
  5540. border: none;
  5541. }
  5542. .url-container textarea:hover,
  5543. .url-container textarea:focus{
  5544. outline: 0;
  5545. box-shadow: 0 0 0 1px #409eff inset;
  5546. }
  5547. `
  5548. )
  5549. });
  5550. let $githubUrlInput = $alert.$shadowRoot.querySelector("#github");
  5551. let $webhookUrlInput = $alert.$shadowRoot.querySelector("#webhook");
  5552. domUtils.on($githubUrlInput, ["input", "propertychange"], (event) => {
  5553. let githubUrl = $githubUrlInput.value.trim();
  5554. let webhookUrlList = [];
  5555. githubUrl.split("\n").forEach((urlStr) => {
  5556. try {
  5557. urlStr = urlStr.trim();
  5558. if (utils.isNull(urlStr)) {
  5559. return;
  5560. }
  5561. let urlObj = new URL(urlStr);
  5562. let urlPathNameSplit = urlObj.pathname.split("/");
  5563. let {
  5564. 1: userName,
  5565. 2: repoName,
  5566. 3: blobStr,
  5567. 4: branchName
  5568. } = urlPathNameSplit;
  5569. let filePath = "";
  5570. if (urlPathNameSplit.length >= 6 && blobStr === "blob") {
  5571. filePath = urlPathNameSplit.slice(5, urlPathNameSplit.length).join("/");
  5572. } else if (urlPathNameSplit.length >= 8 && blobStr === "raw" && branchName === "refs") {
  5573. branchName = urlPathNameSplit[6];
  5574. filePath = urlPathNameSplit.slice(7, urlPathNameSplit.length).join("/");
  5575. } else {
  5576. return;
  5577. }
  5578. if (filePath === "") {
  5579. return;
  5580. }
  5581. webhookUrlList.push(
  5582. `https://raw.githubusercontent.com/${userName}/${repoName}/${branchName}/${filePath}`
  5583. );
  5584. } catch (error) {
  5585. }
  5586. });
  5587. $webhookUrlInput.value = webhookUrlList.join("\n");
  5588. });
  5589. }
  5590. };
  5591. const SettingUIScriptSearch = {
  5592. id: "greasy-fork-panel-config-script-search",
  5593. title: i18next.t("搜索"),
  5594. forms: [
  5595. {
  5596. type: "forms",
  5597. text: "",
  5598. forms: [
  5599. UISwitch(
  5600. i18next.t("新增【{{buttonText}}】按钮", {
  5601. buttonText: i18next.t("名称-全词匹配")
  5602. }),
  5603. "gf-script-search-filterScriptTitleWholeWordMatching",
  5604. true,
  5605. void 0,
  5606. i18next.t("该Checkbox按钮开启后,自动过滤出包含搜索关键词的脚本")
  5607. ),
  5608. UISwitch(
  5609. i18next.t("新增【{{buttonText}}】按钮", {
  5610. buttonText: i18next.t("描述-全词匹配")
  5611. }),
  5612. "gf-script-search-filterScriptDescWholeWordMatching",
  5613. true,
  5614. void 0,
  5615. i18next.t("该Checkbox按钮开启后,自动过滤出包含搜索关键词的脚本")
  5616. ),
  5617. UISwitch(
  5618. i18next.t("新增【{{buttonText}}】按钮", {
  5619. buttonText: i18next.t("作者名称-全词匹配")
  5620. }),
  5621. "gf-script-search-filterScriptAuthorNameWholeWordMatching",
  5622. true,
  5623. void 0,
  5624. i18next.t("该Checkbox按钮开启后,自动过滤出包含搜索关键词的脚本")
  5625. )
  5626. ]
  5627. }
  5628. ]
  5629. };
  5630. const PopsPanel = {
  5631. /** 数据 */
  5632. $data: {
  5633. __data: null,
  5634. __oneSuccessExecMenu: null,
  5635. __onceExec: null,
  5636. __listenData: null,
  5637. /**
  5638. * 菜单项的默认值
  5639. */
  5640. get data() {
  5641. if (PopsPanel.$data.__data == null) {
  5642. PopsPanel.$data.__data = new utils.Dictionary();
  5643. }
  5644. return PopsPanel.$data.__data;
  5645. },
  5646. /**
  5647. * 成功只执行了一次的项
  5648. */
  5649. get oneSuccessExecMenu() {
  5650. if (PopsPanel.$data.__oneSuccessExecMenu == null) {
  5651. PopsPanel.$data.__oneSuccessExecMenu = new utils.Dictionary();
  5652. }
  5653. return PopsPanel.$data.__oneSuccessExecMenu;
  5654. },
  5655. /**
  5656. * 成功只执行了一次的项
  5657. */
  5658. get onceExec() {
  5659. if (PopsPanel.$data.__onceExec == null) {
  5660. PopsPanel.$data.__onceExec = new utils.Dictionary();
  5661. }
  5662. return PopsPanel.$data.__onceExec;
  5663. },
  5664. /** 脚本名,一般用在设置的标题上 */
  5665. get scriptName() {
  5666. return SCRIPT_NAME;
  5667. },
  5668. /** 菜单项的总值在本地数据配置的键名 */
  5669. key: KEY,
  5670. /** 菜单项在attributes上配置的菜单键 */
  5671. attributeKeyName: ATTRIBUTE_KEY,
  5672. /** 菜单项在attributes上配置的菜单默认值 */
  5673. attributeDefaultValueName: ATTRIBUTE_DEFAULT_VALUE
  5674. },
  5675. /** 监听器 */
  5676. $listener: {
  5677. /**
  5678. * 值改变的监听器
  5679. */
  5680. get listenData() {
  5681. if (PopsPanel.$data.__listenData == null) {
  5682. PopsPanel.$data.__listenData = new utils.Dictionary();
  5683. }
  5684. return PopsPanel.$data.__listenData;
  5685. }
  5686. },
  5687. init() {
  5688. this.initPanelDefaultValue();
  5689. this.initExtensionsMenu();
  5690. },
  5691. /** 判断是否是顶层窗口 */
  5692. isTopWindow() {
  5693. return _unsafeWindow.top === _unsafeWindow.self;
  5694. },
  5695. initExtensionsMenu() {
  5696. if (!this.isTopWindow()) {
  5697. return;
  5698. }
  5699. GM_Menu.add([
  5700. {
  5701. key: "show_pops_panel_setting",
  5702. text: i18next.t("⚙ 设置"),
  5703. autoReload: false,
  5704. isStoreValue: false,
  5705. showText(text) {
  5706. return text;
  5707. },
  5708. callback: () => {
  5709. this.showPanel();
  5710. }
  5711. },
  5712. {
  5713. key: "githubUrl2webhookUrl",
  5714. text: "⚙ " + i18next.t("Url To WebhookUrl"),
  5715. autoReload: false,
  5716. isStoreValue: false,
  5717. showText(text) {
  5718. return text;
  5719. },
  5720. callback: () => {
  5721. GithubUrl2WebhookUrl.showView();
  5722. }
  5723. }
  5724. ]);
  5725. },
  5726. /** 初始化菜单项的默认值保存到本地数据中 */
  5727. initPanelDefaultValue() {
  5728. let that = this;
  5729. function initDefaultValue(config) {
  5730. if (!config.attributes) {
  5731. return;
  5732. }
  5733. let needInitConfig = {};
  5734. let key = config.attributes[ATTRIBUTE_KEY];
  5735. if (key != null) {
  5736. needInitConfig[key] = config.attributes[ATTRIBUTE_DEFAULT_VALUE];
  5737. }
  5738. let __attr_init__ = config.attributes[ATTRIBUTE_INIT];
  5739. if (typeof __attr_init__ === "function") {
  5740. let __attr_result__ = __attr_init__();
  5741. if (typeof __attr_result__ === "boolean" && !__attr_result__) {
  5742. return;
  5743. }
  5744. }
  5745. let initMoreValue = config.attributes[ATTRIBUTE_INIT_MORE_VALUE];
  5746. if (initMoreValue && typeof initMoreValue === "object") {
  5747. Object.assign(needInitConfig, initMoreValue);
  5748. }
  5749. let needInitConfigList = Object.keys(needInitConfig);
  5750. if (!needInitConfigList.length) {
  5751. log.warn("请先配置键", config);
  5752. return;
  5753. }
  5754. needInitConfigList.forEach((__key) => {
  5755. let __defaultValue = needInitConfig[__key];
  5756. if (that.$data.data.has(__key)) {
  5757. log.warn("请检查该key(已存在): " + __key);
  5758. }
  5759. that.$data.data.set(__key, __defaultValue);
  5760. });
  5761. }
  5762. function loopInitDefaultValue(configList) {
  5763. for (let index = 0; index < configList.length; index++) {
  5764. let configItem = configList[index];
  5765. initDefaultValue(configItem);
  5766. let childForms = configItem.forms;
  5767. if (childForms && Array.isArray(childForms)) {
  5768. loopInitDefaultValue(childForms);
  5769. }
  5770. }
  5771. }
  5772. let contentConfigList = this.getPanelContentConfig();
  5773. for (let index = 0; index < contentConfigList.length; index++) {
  5774. let leftContentConfigItem = contentConfigList[index];
  5775. if (!leftContentConfigItem.forms) {
  5776. continue;
  5777. }
  5778. let rightContentConfigList = leftContentConfigItem.forms;
  5779. if (rightContentConfigList && Array.isArray(rightContentConfigList)) {
  5780. loopInitDefaultValue(rightContentConfigList);
  5781. }
  5782. }
  5783. },
  5784. /**
  5785. * 设置值
  5786. * @param key 键
  5787. * @param value 值
  5788. */
  5789. setValue(key, value) {
  5790. let locaData = _GM_getValue(KEY, {});
  5791. let oldValue = locaData[key];
  5792. locaData[key] = value;
  5793. _GM_setValue(KEY, locaData);
  5794. if (this.$listener.listenData.has(key)) {
  5795. this.$listener.listenData.get(key).callback(key, oldValue, value);
  5796. }
  5797. },
  5798. /**
  5799. * 获取值
  5800. * @param key 键
  5801. * @param defaultValue 默认值
  5802. */
  5803. getValue(key, defaultValue) {
  5804. let locaData = _GM_getValue(KEY, {});
  5805. let localValue = locaData[key];
  5806. if (localValue == null) {
  5807. if (this.$data.data.has(key)) {
  5808. return this.$data.data.get(key);
  5809. }
  5810. return defaultValue;
  5811. }
  5812. return localValue;
  5813. },
  5814. /**
  5815. * 删除值
  5816. * @param key 键
  5817. */
  5818. deleteValue(key) {
  5819. let locaData = _GM_getValue(KEY, {});
  5820. let oldValue = locaData[key];
  5821. Reflect.deleteProperty(locaData, key);
  5822. _GM_setValue(KEY, locaData);
  5823. if (this.$listener.listenData.has(key)) {
  5824. this.$listener.listenData.get(key).callback(key, oldValue, void 0);
  5825. }
  5826. },
  5827. /**
  5828. * 监听调用setValue、deleteValue
  5829. * @param key 需要监听的键
  5830. * @param callback
  5831. */
  5832. addValueChangeListener(key, callback, option) {
  5833. let listenerId = Math.random();
  5834. this.$listener.listenData.set(key, {
  5835. id: listenerId,
  5836. key,
  5837. callback
  5838. });
  5839. if (option) {
  5840. if (option.immediate) {
  5841. callback(key, this.getValue(key), this.getValue(key));
  5842. }
  5843. }
  5844. return listenerId;
  5845. },
  5846. /**
  5847. * 移除监听
  5848. * @param listenerId 监听的id
  5849. */
  5850. removeValueChangeListener(listenerId) {
  5851. let deleteKey = null;
  5852. for (const [key, value] of this.$listener.listenData.entries()) {
  5853. if (value.id === listenerId) {
  5854. deleteKey = key;
  5855. break;
  5856. }
  5857. }
  5858. if (typeof deleteKey === "string") {
  5859. this.$listener.listenData.delete(deleteKey);
  5860. } else {
  5861. console.warn("没有找到对应的监听器");
  5862. }
  5863. },
  5864. /**
  5865. * 主动触发菜单值改变的回调
  5866. * @param key 菜单键
  5867. * @param newValue 想要触发的新值,默认使用当前值
  5868. * @param oldValue 想要触发的旧值,默认使用当前值
  5869. */
  5870. triggerMenuValueChange(key, newValue, oldValue) {
  5871. if (this.$listener.listenData.has(key)) {
  5872. let listenData = this.$listener.listenData.get(key);
  5873. if (typeof listenData.callback === "function") {
  5874. let value = this.getValue(key);
  5875. let __newValue = value;
  5876. let __oldValue = value;
  5877. if (typeof newValue !== "undefined" && arguments.length > 1) {
  5878. __newValue = newValue;
  5879. }
  5880. if (typeof oldValue !== "undefined" && arguments.length > 2) {
  5881. __oldValue = oldValue;
  5882. }
  5883. listenData.callback(key, __oldValue, __newValue);
  5884. }
  5885. }
  5886. },
  5887. /**
  5888. * 判断该键是否存在
  5889. * @param key 键
  5890. */
  5891. hasKey(key) {
  5892. let locaData = _GM_getValue(KEY, {});
  5893. return key in locaData;
  5894. },
  5895. /**
  5896. * 自动判断菜单是否启用,然后执行回调
  5897. * @param key
  5898. * @param callback 回调
  5899. * @param [isReverse=false] 逆反判断菜单启用
  5900. */
  5901. execMenu(key, callback, isReverse = false) {
  5902. if (!(typeof key === "string" || typeof key === "object" && Array.isArray(key))) {
  5903. throw new TypeError("key 必须是字符串或者字符串数组");
  5904. }
  5905. let runKeyList = [];
  5906. if (typeof key === "object" && Array.isArray(key)) {
  5907. runKeyList = [...key];
  5908. } else {
  5909. runKeyList.push(key);
  5910. }
  5911. let value = void 0;
  5912. for (let index = 0; index < runKeyList.length; index++) {
  5913. const runKey = runKeyList[index];
  5914. if (!this.$data.data.has(runKey)) {
  5915. log.warn(`${key} 键不存在`);
  5916. return;
  5917. }
  5918. let runValue = PopsPanel.getValue(runKey);
  5919. if (isReverse) {
  5920. runValue = !runValue;
  5921. }
  5922. if (!runValue) {
  5923. break;
  5924. }
  5925. value = runValue;
  5926. }
  5927. if (value) {
  5928. callback(value);
  5929. }
  5930. },
  5931. /**
  5932. * 自动判断菜单是否启用,然后执行回调,只会执行一次
  5933. * @param key
  5934. * @param callback 回调
  5935. * @param getValueFn 自定义处理获取当前值,值true是启用并执行回调,值false是不执行回调
  5936. * @param handleValueChangeFn 自定义处理值改变时的回调,值true是启用并执行回调,值false是不执行回调
  5937. */
  5938. execMenuOnce(key, callback, getValueFn, handleValueChangeFn) {
  5939. if (typeof key !== "string") {
  5940. throw new TypeError("key 必须是字符串");
  5941. }
  5942. if (!this.$data.data.has(key)) {
  5943. log.warn(`${key} 键不存在`);
  5944. return;
  5945. }
  5946. if (this.$data.oneSuccessExecMenu.has(key)) {
  5947. return;
  5948. }
  5949. this.$data.oneSuccessExecMenu.set(key, 1);
  5950. let __getValue = () => {
  5951. let localValue = PopsPanel.getValue(key);
  5952. return typeof getValueFn === "function" ? getValueFn(key, localValue) : localValue;
  5953. };
  5954. let resultStyleList = [];
  5955. let dynamicPushStyleNode = ($style) => {
  5956. let __value = __getValue();
  5957. let dynamicResultList = [];
  5958. if ($style instanceof HTMLStyleElement) {
  5959. dynamicResultList = [$style];
  5960. } else if (Array.isArray($style)) {
  5961. dynamicResultList = [
  5962. ...$style.filter(
  5963. (item) => item != null && item instanceof HTMLStyleElement
  5964. )
  5965. ];
  5966. }
  5967. if (__value) {
  5968. resultStyleList = resultStyleList.concat(dynamicResultList);
  5969. } else {
  5970. for (let index = 0; index < dynamicResultList.length; index++) {
  5971. let $css = dynamicResultList[index];
  5972. $css.remove();
  5973. dynamicResultList.splice(index, 1);
  5974. index--;
  5975. }
  5976. }
  5977. };
  5978. let changeCallBack = (currentValue) => {
  5979. let resultList = [];
  5980. if (currentValue) {
  5981. let result = callback(currentValue, dynamicPushStyleNode);
  5982. if (result instanceof HTMLStyleElement) {
  5983. resultList = [result];
  5984. } else if (Array.isArray(result)) {
  5985. resultList = [
  5986. ...result.filter(
  5987. (item) => item != null && item instanceof HTMLStyleElement
  5988. )
  5989. ];
  5990. }
  5991. }
  5992. for (let index = 0; index < resultStyleList.length; index++) {
  5993. let $css = resultStyleList[index];
  5994. $css.remove();
  5995. resultStyleList.splice(index, 1);
  5996. index--;
  5997. }
  5998. resultStyleList = [...resultList];
  5999. };
  6000. this.addValueChangeListener(
  6001. key,
  6002. (__key, oldValue, newValue) => {
  6003. let __newValue = newValue;
  6004. if (typeof handleValueChangeFn === "function") {
  6005. __newValue = handleValueChangeFn(__key, newValue, oldValue);
  6006. }
  6007. changeCallBack(__newValue);
  6008. }
  6009. );
  6010. let value = __getValue();
  6011. if (value) {
  6012. changeCallBack(value);
  6013. }
  6014. },
  6015. /**
  6016. * 父子菜单联动,自动判断菜单是否启用,然后执行回调,只会执行一次
  6017. * @param key 菜单键
  6018. * @param childKey 子菜单键
  6019. * @param callback 回调
  6020. * @param replaceValueFn 用于修改mainValue,返回undefined则不做处理
  6021. */
  6022. execInheritMenuOnce(key, childKey, callback, replaceValueFn) {
  6023. let that = this;
  6024. const handleInheritValue = (key2, childKey2) => {
  6025. let mainValue = that.getValue(key2);
  6026. let childValue = that.getValue(childKey2);
  6027. if (typeof replaceValueFn === "function") {
  6028. let changedMainValue = replaceValueFn(mainValue, childValue);
  6029. if (changedMainValue !== void 0) {
  6030. return changedMainValue;
  6031. }
  6032. }
  6033. return mainValue;
  6034. };
  6035. this.execMenuOnce(
  6036. key,
  6037. callback,
  6038. () => {
  6039. return handleInheritValue(key, childKey);
  6040. },
  6041. () => {
  6042. return handleInheritValue(key, childKey);
  6043. }
  6044. );
  6045. this.execMenuOnce(
  6046. childKey,
  6047. () => {
  6048. },
  6049. () => false,
  6050. () => {
  6051. this.triggerMenuValueChange(key);
  6052. return false;
  6053. }
  6054. );
  6055. },
  6056. /**
  6057. * 根据key执行一次
  6058. * @param key
  6059. */
  6060. onceExec(key, callback) {
  6061. if (typeof key !== "string") {
  6062. throw new TypeError("key 必须是字符串");
  6063. }
  6064. if (this.$data.onceExec.has(key)) {
  6065. return;
  6066. }
  6067. callback();
  6068. this.$data.onceExec.set(key, 1);
  6069. },
  6070. /**
  6071. * 显示设置面板
  6072. */
  6073. showPanel() {
  6074. __pops.panel({
  6075. title: {
  6076. text: i18next.t("{{SCRIPT_NAME}}-设置", { SCRIPT_NAME }),
  6077. position: "center",
  6078. html: false,
  6079. style: ""
  6080. },
  6081. content: this.getPanelContentConfig(),
  6082. mask: {
  6083. enable: true,
  6084. clickEvent: {
  6085. toClose: true,
  6086. toHide: false
  6087. }
  6088. },
  6089. zIndex() {
  6090. let maxZIndex = Utils.getMaxZIndex();
  6091. let popsMaxZIndex = __pops.config.InstanceUtils.getPopsMaxZIndex().zIndex;
  6092. return Utils.getMaxValue(maxZIndex, popsMaxZIndex) + 100;
  6093. },
  6094. width: PanelUISize.setting.width,
  6095. height: PanelUISize.setting.height,
  6096. drag: true,
  6097. only: true,
  6098. style: `
  6099. ${UIScriptListCSS}
  6100. `
  6101. });
  6102. },
  6103. /**
  6104. * 获取配置内容
  6105. */
  6106. getPanelContentConfig() {
  6107. let configList = [
  6108. SettingUICommon,
  6109. SettingUIScripts,
  6110. SettingUIScriptSearch,
  6111. SettingUIDiscuessions,
  6112. SettingUIUsers
  6113. ];
  6114. return configList;
  6115. }
  6116. };
  6117. const beautifyMarkdownCSS = ':root {\r\n --borderColor-muted: #d1d9e0b3;\r\n}\r\ncode {\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\r\n/* 上面是通用 */\r\n\r\n/* 下面是自定义 */\r\n.user-content {\r\n background: #ffffff !important;\r\n border-left-color: #ffffff !important;\r\n}\r\n.user-content a {\r\n color: #0969da;\r\n}\r\n.user-content h1 {\r\n padding-bottom: 0.3em;\r\n font-size: 2em;\r\n border-bottom: 1px solid var(--borderColor-muted, var(--color-border-muted));\r\n}\r\n.user-content h2 {\r\n padding-bottom: 0.3em;\r\n font-size: 1.5em;\r\n border-bottom: 1px solid var(--borderColor-muted, var(--color-border-muted));\r\n}\r\n/* 任务列表样式 */\r\n.task-list-item-checkbox {\r\n margin: 0 0.2em 0.25em -1.4em;\r\n vertical-align: middle;\r\n}\r\n/* 隐藏标记框 */\r\nul li:has(> .task-list-item-checkbox)::marker {\r\n content: "";\r\n}\r\n\r\n/* 警告列表 */\r\n.markdown-alert {\r\n --borderColor-default: #d1d9e0;\r\n padding: 0.5rem 1rem;\r\n margin-bottom: 1rem;\r\n color: inherit;\r\n border-left: 0.25em solid var(--borderColor-default);\r\n border-left-color: var(--fgColor-accent);\r\n}\r\n.markdown-alert-title {\r\n display: flex;\r\n font-weight: 500;\r\n align-items: center;\r\n line-height: 1;\r\n color: var(--fgColor-accent);\r\n}\r\n.markdown-alert > :last-child {\r\n margin-bottom: 0;\r\n}\r\n.markdown-alert-title .octicon {\r\n display: inline-block;\r\n overflow: visible !important;\r\n vertical-align: text-bottom;\r\n fill: currentColor;\r\n}\r\n.markdown-alert-title .mr-2 {\r\n margin-right: 0.5rem !important;\r\n}\r\n.markdown-alert[data-type="NOTE"] {\r\n --fgColor-accent: #0969da;\r\n}\r\n\r\n.markdown-alert[data-type="TIP"] {\r\n --fgColor-accent: #1a7f37;\r\n}\r\n.markdown-alert[data-type="IMPORTANT"] {\r\n --fgColor-accent: #8250df;\r\n}\r\n.markdown-alert[data-type="WARNING"] {\r\n --fgColor-accent: #9a6700;\r\n}\r\n.markdown-alert[data-type="CAUTION"] {\r\n --fgColor-accent: #d1242f;\r\n}\r\n';
  6118. const beautifyButtonCSS = '/* 美化按钮 */\r\ninput[type="submit"],\r\ninput[type="button"],\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\ninput[type="button"]:hover,\r\ninput[type="button"]: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';
  6119. 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';
  6120. 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';
  6121. 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";
  6122. 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';
  6123. 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";
  6124. const beautifyTopNavigationBarCSS = "#language-selector {\r\n display: none;\r\n}\r\n/* PC端 */\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 body > .width-constraint {\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 flex: 1 0 auto;\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/* mobile端 */\r\n@media screen and (max-width: 600px) {\r\n header#main-header #site-name-text h1 {\r\n line-height: normal;\r\n padding-bottom: 0;\r\n }\r\n /* 美化移动端顶部导航栏的更多 */\r\n #mobile-nav nav {\r\n font-size: 1rem !important;\r\n }\r\n /* 添加鼠标悬浮效果 */\r\n #mobile-nav nav li:hover {\r\n background: #840404;\r\n }\r\n /* 去除链接下划线 */\r\n #mobile-nav nav li a {\r\n text-decoration: none;\r\n }\r\n}\r\n";
  6125. const beautifyHomeCSS = '.browser-list li::before,\r\n.browser-list-selector::before {\r\n content: "";\r\n width: 22px;\r\n height: 22px;\r\n display: inline-block;\r\n vertical-align: -4px;\r\n margin: 0px 5px;\r\n background-size: 100% 100% !important;\r\n background-repeat: no-repeat !important;\r\n}\r\n/* Desktop */\r\n.browser-list-selector[data-for="desktop-browser-list"]::before {\r\n background: url("");\r\n}\r\n/* Android */\r\n.browser-list-selector[data-for="android-browser-list"]::before {\r\n background: url("");\r\n}\r\n/* iOS */\r\n.browser-list-selector[data-for="ios-browser-list"]::before {\r\n background: url("");\r\n}\r\n/* Chrome */\r\n#desktop-browser-list li:nth-child(1)::before {\r\n background: url("");\r\n}\r\n/* Firefox */\r\n#desktop-browser-list li:nth-child(2)::before,\r\n#android-browser-list li:nth-child(1)::before {\r\n background: url("");\r\n}\r\n/* Safari */\r\n#desktop-browser-list li:nth-child(3)::before,\r\n#ios-browser-list li:nth-child(1)::before {\r\n background: url("");\r\n}\r\n/* Edge */\r\n#desktop-browser-list li:nth-child(4)::before {\r\n background: url("");\r\n}\r\n/* Opera */\r\n#desktop-browser-list li:nth-child(5)::before {\r\n background: url("");\r\n}\r\n/* Maxthon */\r\n#desktop-browser-list li:nth-child(6)::before,\r\n#android-browser-list li:nth-child(2)::before {\r\n background: url("");\r\n}\r\n/* AdGuard */\r\n#desktop-browser-list li:nth-child(7)::before {\r\n background: url("");\r\n}\r\n/* Dolphin */\r\n#android-browser-list li:nth-child(3)::before {\r\n background: url("");\r\n}\r\n/* UC */\r\n#android-browser-list li:nth-child(4)::before {\r\n background: url("");\r\n}\r\n/* Kiwi */\r\n#android-browser-list li:nth-child(5)::before {\r\n background: url("");\r\n}\r\n/* X浏览器 */\r\n#android-browser-list li:nth-child(6)::before {\r\n background: url("");\r\n}\r\n/* 书签地球 */\r\n#android-browser-list li:nth-child(7)::before {\r\n background: url("");\r\n}\r\n/* M浏览器 */\r\n#android-browser-list li:nth-child(8)::before {\r\n background: url("");\r\n}\r\n/* 狐猴浏览器 */\r\n#android-browser-list li:nth-child(9)::before {\r\n background: url("");\r\n}\r\n/* Gear */\r\n#ios-browser-list li:nth-child(2)::before {\r\n background: url("");\r\n}\r\n';
  6126. const beautifyFeedbackCSS = "";
  6127. const GreasyforkBeautify = {
  6128. init() {
  6129. PopsPanel.execMenuOnce("beautifyPage", () => {
  6130. return this.beautifyPageElement();
  6131. });
  6132. PopsPanel.execMenuOnce("beautifyGreasyforkBeautify", () => {
  6133. return this.beautifyGreasyforkBeautify();
  6134. });
  6135. PopsPanel.execMenuOnce("beautifyUploadImage", () => {
  6136. return this.beautifyUploadImage();
  6137. });
  6138. PopsPanel.execMenuOnce("beautifyTopNavigationBar", () => {
  6139. return this.beautifyTopNavigationBar();
  6140. });
  6141. },
  6142. /**
  6143. * 美化页面元素,包括Markdown
  6144. */
  6145. beautifyPageElement() {
  6146. log.info("美化页面元素");
  6147. let result = [];
  6148. result.push(addStyle(beautifyMarkdownCSS));
  6149. result.push(addStyle(beautifyButtonCSS));
  6150. result.push(addStyle(beautifyRadioCSS));
  6151. result.push(addStyle(beautifyInputCSS));
  6152. result.push(addStyle(beautifyTextAreaCSS));
  6153. result.push(
  6154. addStyle(
  6155. /*css*/
  6156. `
  6157. p:has(input[type="submit"][name="update-and-sync"]){
  6158. margin-top: 10px;
  6159. }
  6160. `
  6161. )
  6162. );
  6163. domUtils.ready(() => {
  6164. let $markupChoice = $(
  6165. 'a[target="markup_choice"][href*="daringfireball.net"]'
  6166. );
  6167. if ($markupChoice) {
  6168. $markupChoice.parentElement.replaceChild(
  6169. domUtils.createElement("span", {
  6170. textContent: "Markdown"
  6171. }),
  6172. $markupChoice
  6173. );
  6174. }
  6175. $$(".user-content ul li").forEach(($li) => {
  6176. var _a2, _b;
  6177. let $first = $li.firstChild;
  6178. if (($first == null ? void 0 : $first.nodeName) === "#text") {
  6179. if ((_a2 = $first.textContent) == null ? void 0 : _a2.startsWith("[x] ")) {
  6180. $first.textContent = $first.textContent.replace("[x] ", "");
  6181. domUtils.prepend(
  6182. $li,
  6183. /*html*/
  6184. `
  6185. <input type="checkbox" disabled="" class="task-list-item-checkbox" checked="">
  6186. `
  6187. );
  6188. } else if ((_b = $first.textContent) == null ? void 0 : _b.startsWith("[ ] ")) {
  6189. $first.textContent = $first.textContent.replace("[ ] ", "");
  6190. domUtils.prepend(
  6191. $li,
  6192. /*html*/
  6193. `
  6194. <input type="checkbox" disabled="" class="task-list-item-checkbox">
  6195. `
  6196. );
  6197. }
  6198. }
  6199. });
  6200. let warningListMap = {
  6201. "[!NOTE]": (
  6202. /*html*/
  6203. `
  6204. <p class="markdown-alert-title" data-type="NOTE">
  6205. <svg class="octicon octicon-info mr-2" viewBox="0 0 16 16" version="1.1" width="16" height="16">
  6206. <path d="M0 8a8 8 0 1 1 16 0A8 8 0 0 1 0 8Zm8-6.5a6.5 6.5 0 1 0 0 13 6.5 6.5 0 0 0 0-13ZM6.5 7.75A.75.75 0 0 1 7.25 7h1a.75.75 0 0 1 .75.75v2.75h.25a.75.75 0 0 1 0 1.5h-2a.75.75 0 0 1 0-1.5h.25v-2h-.25a.75.75 0 0 1-.75-.75ZM8 6a1 1 0 1 1 0-2 1 1 0 0 1 0 2Z"></path>
  6207. </svg>Note
  6208. </p>
  6209. `
  6210. ),
  6211. "[!TIP]": (
  6212. /*html*/
  6213. `
  6214. <p class="markdown-alert-title" data-type="TIP">
  6215. <svg class="octicon octicon-light-bulb mr-2" viewBox="0 0 16 16" version="1.1" width="16" height="16" aria-hidden="true">
  6216. <path d="M8 1.5c-2.363 0-4 1.69-4 3.75 0 .984.424 1.625.984 2.304l.214.253c.223.264.47.556.673.848.284.411.537.896.621 1.49a.75.75 0 0 1-1.484.211c-.04-.282-.163-.547-.37-.847a8.456 8.456 0 0 0-.542-.68c-.084-.1-.173-.205-.268-.32C3.201 7.75 2.5 6.766 2.5 5.25 2.5 2.31 4.863 0 8 0s5.5 2.31 5.5 5.25c0 1.516-.701 2.5-1.328 3.259-.095.115-.184.22-.268.319-.207.245-.383.453-.541.681-.208.3-.33.565-.37.847a.751.751 0 0 1-1.485-.212c.084-.593.337-1.078.621-1.489.203-.292.45-.584.673-.848.075-.088.147-.173.213-.253.561-.679.985-1.32.985-2.304 0-2.06-1.637-3.75-4-3.75ZM5.75 12h4.5a.75.75 0 0 1 0 1.5h-4.5a.75.75 0 0 1 0-1.5ZM6 15.25a.75.75 0 0 1 .75-.75h2.5a.75.75 0 0 1 0 1.5h-2.5a.75.75 0 0 1-.75-.75Z"></path>
  6217. </svg>Tip
  6218. </p>
  6219. `
  6220. ),
  6221. "[!IMPORTANT]": (
  6222. /*html*/
  6223. `
  6224. <p class="markdown-alert-title" data-type="IMPORTANT">
  6225. <svg class="octicon octicon-report mr-2" viewBox="0 0 16 16" version="1.1" width="16" height="16" aria-hidden="true"><path d="M0 1.75C0 .784.784 0 1.75 0h12.5C15.216 0 16 .784 16 1.75v9.5A1.75 1.75 0 0 1 14.25 13H8.06l-2.573 2.573A1.458 1.458 0 0 1 3 14.543V13H1.75A1.75 1.75 0 0 1 0 11.25Zm1.75-.25a.25.25 0 0 0-.25.25v9.5c0 .138.112.25.25.25h2a.75.75 0 0 1 .75.75v2.19l2.72-2.72a.749.749 0 0 1 .53-.22h6.5a.25.25 0 0 0 .25-.25v-9.5a.25.25 0 0 0-.25-.25Zm7 2.25v2.5a.75.75 0 0 1-1.5 0v-2.5a.75.75 0 0 1 1.5 0ZM9 9a1 1 0 1 1-2 0 1 1 0 0 1 2 0Z"></path>
  6226. </svg>Important
  6227. </p>
  6228. `
  6229. ),
  6230. "[!WARNING]": (
  6231. /*html*/
  6232. `
  6233. <p class="markdown-alert-title" data-type="WARNING">
  6234. <svg class="octicon octicon-alert mr-2" viewBox="0 0 16 16" version="1.1" width="16" height="16" aria-hidden="true"><path d="M6.457 1.047c.659-1.234 2.427-1.234 3.086 0l6.082 11.378A1.75 1.75 0 0 1 14.082 15H1.918a1.75 1.75 0 0 1-1.543-2.575Zm1.763.707a.25.25 0 0 0-.44 0L1.698 13.132a.25.25 0 0 0 .22.368h12.164a.25.25 0 0 0 .22-.368Zm.53 3.996v2.5a.75.75 0 0 1-1.5 0v-2.5a.75.75 0 0 1 1.5 0ZM9 11a1 1 0 1 1-2 0 1 1 0 0 1 2 0Z"></path>
  6235. </svg>Warning
  6236. </p>
  6237. `
  6238. ),
  6239. "[!CAUTION]": (
  6240. /*html*/
  6241. `
  6242. <p class="markdown-alert-title" data-type="CAUTION">
  6243. <svg class="octicon octicon-stop mr-2" viewBox="0 0 16 16" version="1.1" width="16" height="16" aria-hidden="true"><path d="M4.47.22A.749.749 0 0 1 5 0h6c.199 0 .389.079.53.22l4.25 4.25c.141.14.22.331.22.53v6a.749.749 0 0 1-.22.53l-4.25 4.25A.749.749 0 0 1 11 16H5a.749.749 0 0 1-.53-.22L.22 11.53A.749.749 0 0 1 0 11V5c0-.199.079-.389.22-.53Zm.84 1.28L1.5 5.31v5.38l3.81 3.81h5.38l3.81-3.81V5.31L10.69 1.5ZM8 4a.75.75 0 0 1 .75.75v3.5a.75.75 0 0 1-1.5 0v-3.5A.75.75 0 0 1 8 4Zm0 8a1 1 0 1 1 0-2 1 1 0 0 1 0 2Z"></path>
  6244. </svg>Caution
  6245. </p>
  6246. `
  6247. )
  6248. };
  6249. let warningList = Object.keys(warningListMap);
  6250. $$(".user-content blockquote").forEach(($blockQuote) => {
  6251. Array.from($blockQuote.children).forEach(($child) => {
  6252. let htmlStr = domUtils.html($child);
  6253. if ($child.nodeName === "P") {
  6254. let findValue = warningList.find((value) => {
  6255. return htmlStr.toLowerCase().startsWith(value.toLowerCase());
  6256. });
  6257. if (findValue) {
  6258. domUtils.html(
  6259. $child,
  6260. htmlStr.replace(findValue, "")
  6261. );
  6262. domUtils.before(
  6263. $child,
  6264. warningListMap[findValue]
  6265. );
  6266. if (domUtils.html($child) === "") {
  6267. $child.remove();
  6268. }
  6269. }
  6270. }
  6271. });
  6272. });
  6273. $$(".user-content > blockquote").forEach(($blockQuote) => {
  6274. Array.from(
  6275. $blockQuote.querySelectorAll(".markdown-alert-title")
  6276. ).forEach(($markdownAlertTitle) => {
  6277. var _a2, _b;
  6278. let $markdownAlert = domUtils.createElement(
  6279. "div",
  6280. {
  6281. className: "markdown-alert markdown-alert-tip"
  6282. },
  6283. {
  6284. "data-type": $markdownAlertTitle.getAttribute("data-type")
  6285. }
  6286. );
  6287. while ($markdownAlertTitle.nextSibling) {
  6288. if ((_b = (_a2 = $markdownAlertTitle.nextSibling) == null ? void 0 : _a2.classList) == null ? void 0 : _b.contains("markdown-alert-title")) {
  6289. break;
  6290. }
  6291. domUtils.append(
  6292. $markdownAlert,
  6293. $markdownAlertTitle.nextSibling
  6294. );
  6295. }
  6296. domUtils.prepend($markdownAlert, $markdownAlertTitle);
  6297. domUtils.before($blockQuote, $markdownAlert);
  6298. });
  6299. });
  6300. if (GreasyforkRouter.isHome()) {
  6301. result.push(addStyle(beautifyHomeCSS));
  6302. } else if (GreasyforkRouter.isScriptsFeedback()) {
  6303. result.push(addStyle(beautifyFeedbackCSS));
  6304. let $noRating = document.querySelector(
  6305. '.radio-label[for*="discussion_rating_0"]'
  6306. );
  6307. if ($noRating) {
  6308. $noRating.innerHTML = $noRating.innerHTML.replace(
  6309. "不评分",
  6310. '<span class="rating-icon rating-icon-none">不评分</span>'
  6311. );
  6312. }
  6313. let $badRating = document.querySelector(
  6314. '.radio-label[for*="discussion_rating_2"]'
  6315. );
  6316. if ($badRating) {
  6317. $badRating.innerHTML = $badRating.innerHTML.replace(
  6318. "差评",
  6319. '<span class="rating-icon rating-icon-bad">差评</span>'
  6320. );
  6321. }
  6322. let $okRating = document.querySelector(
  6323. '.radio-label[for*="discussion_rating_3"]'
  6324. );
  6325. if ($okRating) {
  6326. $okRating.innerHTML = $okRating.innerHTML.replace(
  6327. "一般",
  6328. '<span class="rating-icon rating-icon-ok">一般</span>'
  6329. );
  6330. }
  6331. let $goodRating = document.querySelector(
  6332. '.radio-label[for*="discussion_rating_4"]'
  6333. );
  6334. if ($goodRating) {
  6335. $goodRating.innerHTML = $goodRating.innerHTML.replace(
  6336. "好评",
  6337. '<span class="rating-icon rating-icon-good">好评</span>'
  6338. );
  6339. }
  6340. } else if (GreasyforkRouter.isScriptAdmin()) {
  6341. if (!document.querySelector('input[type="submit"][name="update-only"]')) {
  6342. result.push(
  6343. addStyle(
  6344. /*css*/
  6345. `
  6346. .indented{
  6347. padding-left: unset;
  6348. }
  6349. `
  6350. )
  6351. );
  6352. }
  6353. }
  6354. });
  6355. return result;
  6356. },
  6357. /**
  6358. * 美化 Greasyfork Beautify脚本
  6359. */
  6360. beautifyGreasyforkBeautify() {
  6361. log.info("美化 Greasyfork Beautify脚本");
  6362. let result = [];
  6363. result.push(addStyle(compatibleBeautifyCSS));
  6364. if (utils.isPhone()) {
  6365. result.push(
  6366. addStyle(
  6367. /*css*/
  6368. `
  6369. section#script-info,
  6370. section.text-content,
  6371. div.width-constraint table.text-content.log-table{
  6372. margin-top: 80px;
  6373. }
  6374. div.width-constraint div.sidebarred{
  6375. padding-top: 80px;
  6376. }
  6377. div.width-constraint div.sidebarred .sidebar{
  6378. top: 80px;
  6379. }`
  6380. )
  6381. );
  6382. } else {
  6383. result.push(
  6384. addStyle(
  6385. /*css*/
  6386. `
  6387. section#script-info{
  6388. margin-top: 10px;
  6389. }`
  6390. )
  6391. );
  6392. }
  6393. return result;
  6394. },
  6395. /**
  6396. * 美化上传图片
  6397. */
  6398. beautifyUploadImage() {
  6399. log.info("美化上传图片");
  6400. let result = [];
  6401. result.push(addStyle(beautifyUploadImageCSS));
  6402. domUtils.ready(() => {
  6403. function clearErrorTip(element) {
  6404. while (element.nextElementSibling) {
  6405. element.parentElement.removeChild(element.nextElementSibling);
  6406. }
  6407. }
  6408. let $fileInputList = document.querySelectorAll('input[type="file"]');
  6409. $fileInputList.forEach(($input) => {
  6410. if ($input.getAttribute("name") === "code_upload") {
  6411. return;
  6412. }
  6413. if ($input.hasAttribute("accept") && $input.getAttribute("accept").includes("javascript")) {
  6414. return;
  6415. }
  6416. domUtils.on($input, ["propertychange", "input"], function(event) {
  6417. clearErrorTip(event.target);
  6418. let chooseImageFiles = event.currentTarget.files;
  6419. if (!chooseImageFiles) {
  6420. return;
  6421. }
  6422. if (chooseImageFiles.length === 0) {
  6423. return;
  6424. }
  6425. log.info("选择的图片", chooseImageFiles);
  6426. if (chooseImageFiles.length > 5) {
  6427. domUtils.after(
  6428. $input,
  6429. domUtils.createElement("p", {
  6430. textContent: i18next.t(`❌ 最多同时长传5张图片`)
  6431. })
  6432. );
  6433. }
  6434. let notAllowImage = [];
  6435. Array.from(chooseImageFiles).forEach((imageFile) => {
  6436. if (imageFile.size > 204800 || !imageFile.type.match(/png|jpg|jpeg|gif|apng|webp/i)) {
  6437. notAllowImage.push(imageFile);
  6438. }
  6439. });
  6440. if (notAllowImage.length === 0) {
  6441. return;
  6442. }
  6443. notAllowImage.forEach((imageFile) => {
  6444. domUtils.after(
  6445. $input,
  6446. domUtils.createElement("p", {
  6447. textContent: i18next.t("❌ 图片:{{name}} 大小:{{size}}", {
  6448. name: imageFile.name,
  6449. size: imageFile.size
  6450. })
  6451. })
  6452. );
  6453. });
  6454. });
  6455. });
  6456. let $textAreaSelectorString = [
  6457. /* 某条反馈内的回复框 */
  6458. "textarea#comment_text",
  6459. /* 反馈页面的回复框 */
  6460. "textarea.comment-entry"
  6461. ];
  6462. $textAreaSelectorString.forEach((selector) => {
  6463. domUtils.on(selector, "paste", (event) => {
  6464. log.info("触发粘贴事件", event);
  6465. setTimeout(() => {
  6466. domUtils.trigger($fileInputList, "input");
  6467. }, 100);
  6468. });
  6469. });
  6470. });
  6471. return result;
  6472. },
  6473. /**
  6474. * 美化顶部导航栏
  6475. */
  6476. beautifyTopNavigationBar() {
  6477. log.info("美化顶部导航栏");
  6478. let result = [];
  6479. result.push(addStyle(beautifyTopNavigationBarCSS));
  6480. if (window.outerWidth > 550) {
  6481. result.push(CommonUtil.addBlockCSS(".with-submenu"));
  6482. domUtils.ready(() => {
  6483. let $siteNav = $("#site-nav");
  6484. let $siteNavNav = $siteNav.querySelector("nav");
  6485. $$(".with-submenu nav li").forEach(($ele) => {
  6486. $siteNavNav.appendChild($ele);
  6487. });
  6488. });
  6489. }
  6490. return result;
  6491. }
  6492. };
  6493. const GreasyforkAccount = {
  6494. init() {
  6495. PopsPanel.execMenu("autoLogin", () => {
  6496. this.autoLogin();
  6497. });
  6498. },
  6499. /**
  6500. * 自动登录
  6501. */
  6502. autoLogin() {
  6503. utils.waitNode("span.sign-in-link a[rel=nofollow]").then(async () => {
  6504. let user = PopsPanel.getValue("user");
  6505. let pwd = PopsPanel.getValue("pwd");
  6506. if (utils.isNull(user)) {
  6507. Qmsg.error(i18next.t("请先在菜单中录入账号"));
  6508. return;
  6509. }
  6510. if (utils.isNull(pwd)) {
  6511. Qmsg.error(i18next.t("请先在菜单中录入密码"));
  6512. return;
  6513. }
  6514. let csrfToken = document.querySelector("meta[name='csrf-token']");
  6515. if (!csrfToken) {
  6516. Qmsg.error(i18next.t("获取csrf-token失败"));
  6517. return;
  6518. }
  6519. let loginTip = Qmsg.loading(i18next.t("正在登录中..."));
  6520. let postResp = await httpx.post(
  6521. "https://greasyfork.org/zh-CN/users/sign_in",
  6522. {
  6523. fetch: true,
  6524. data: encodeURI(
  6525. `authenticity_token=${csrfToken.getAttribute(
  6526. "content"
  6527. )}&user[email]=${user}&user[password]=${pwd}&user[remember_me]=1&commit=登录`
  6528. ),
  6529. headers: {
  6530. "Content-Type": "application/x-www-form-urlencoded"
  6531. }
  6532. }
  6533. );
  6534. loginTip.destroy();
  6535. if (!postResp.status) {
  6536. log.error(postResp);
  6537. Qmsg.error(i18next.t("登录失败,请在控制台查看原因"));
  6538. return;
  6539. }
  6540. let respText = postResp.data.responseText;
  6541. let parseLoginHTMLNode = domUtils.parseHTML(respText, true, true);
  6542. if (parseLoginHTMLNode.querySelectorAll(
  6543. ".sign-out-link a[rel=nofollow][data-method='delete']"
  6544. ).length) {
  6545. Qmsg.success(i18next.t("登录成功,1s后自动跳转"));
  6546. setTimeout(() => {
  6547. window.location.reload();
  6548. }, 1e3);
  6549. } else {
  6550. log.error(postResp);
  6551. log.error(`当前账号:${user}`);
  6552. log.error(`当前密码:${pwd}`);
  6553. Qmsg.error(
  6554. i18next.t("登录失败,可能是账号/密码错误,请在控制台查看原因")
  6555. );
  6556. }
  6557. });
  6558. }
  6559. };
  6560. const GreasyforkForum = {
  6561. init() {
  6562. this.readBgColor();
  6563. domUtils.ready(() => {
  6564. PopsPanel.execMenuOnce("greasyfork-discussions-filter-enable", () => {
  6565. this.filterEnable();
  6566. });
  6567. PopsPanel.execMenuOnce("discussions-addShortcutOperationButton", () => {
  6568. this.addShortcutOperationButton();
  6569. });
  6570. PopsPanel.execMenuOnce("discussions-addReportButton", () => {
  6571. this.addReportButton();
  6572. });
  6573. });
  6574. },
  6575. /**
  6576. * 启用Greasyfork论坛过滤器
  6577. */
  6578. filterEnable() {
  6579. log.info("启用Greasyfork论坛过滤器");
  6580. GreasyforkDiscussionsFilter.init();
  6581. },
  6582. /**
  6583. * 设置已读背景颜色
  6584. */
  6585. readBgColor() {
  6586. log.info("设置已读背景颜色");
  6587. let color = PopsPanel.getValue("discussions-readBgColor");
  6588. return addStyle(
  6589. /*css*/
  6590. `
  6591. .discussion-read{
  6592. background: ${color} !important;
  6593. }
  6594. `
  6595. );
  6596. },
  6597. /**
  6598. * 添加【过滤】按钮
  6599. */
  6600. addShortcutOperationButton() {
  6601. log.info("添加【过滤】按钮");
  6602. GreasyforkDiscussionsFilter.getElementList().forEach(($listContainer) => {
  6603. if ($listContainer.querySelector(
  6604. ".discussion-filter-button"
  6605. )) {
  6606. return;
  6607. }
  6608. let $listItem = $listContainer.querySelector(
  6609. ".discussion-list-item"
  6610. );
  6611. let $meta = $listItem.querySelector(".discussion-meta");
  6612. let $ownMetaItem = domUtils.createElement(
  6613. "div",
  6614. {
  6615. className: "discussion-meta-item",
  6616. innerHTML: `
  6617. <button class="discussion-filter-button">${i18next.t("过滤")}</button>
  6618. `
  6619. },
  6620. {
  6621. style: "flex: 0;"
  6622. }
  6623. );
  6624. let $button = $ownMetaItem.querySelector(
  6625. ".discussion-filter-button"
  6626. );
  6627. $meta.appendChild($ownMetaItem);
  6628. domUtils.on($button, "click", (event) => {
  6629. var _a2, _b, _c;
  6630. utils.preventEvent(event);
  6631. const discussionInfo = GreasyforkDiscussionsFilter.parseDiscuessionListContainerInfo(
  6632. $listContainer
  6633. );
  6634. let attr_filter_key = "data-filter-key";
  6635. let attr_filter_value = "data-filter-value";
  6636. let $dialog = __pops.alert({
  6637. title: {
  6638. text: i18next.t("选择需要过滤的选项"),
  6639. position: "center",
  6640. html: false
  6641. },
  6642. content: {
  6643. text: (
  6644. /*html*/
  6645. `
  6646. <button ${attr_filter_key}="scriptId" ${attr_filter_value}="^${discussionInfo.scriptId}$">${i18next.t("脚本id:{{text}}", {
  6647. text: discussionInfo.scriptId
  6648. })}</button>
  6649. <button ${attr_filter_key}="scriptName" ${attr_filter_value}="^${utils.parseStringToRegExpString(
  6650. discussionInfo.scriptName
  6651. )}$">${i18next.t("脚本名:{{text}}", {
  6652. text: discussionInfo.scriptName
  6653. })}</button>
  6654. <button ${attr_filter_key}="postUserId" ${attr_filter_value}="^${utils.parseStringToRegExpString(
  6655. discussionInfo.postUserId
  6656. )}$">${i18next.t("发布的用户id:{{text}}", {
  6657. text: discussionInfo.postUserId
  6658. })}</button>
  6659. `
  6660. ),
  6661. html: true
  6662. },
  6663. mask: {
  6664. enable: true,
  6665. clickEvent: {
  6666. toClose: true
  6667. }
  6668. },
  6669. btn: {
  6670. ok: {
  6671. enable: false
  6672. }
  6673. },
  6674. drag: true,
  6675. dragLimit: true,
  6676. width: "350px",
  6677. height: "300px",
  6678. style: (
  6679. /*css*/
  6680. `
  6681. .pops-alert-content{
  6682. display: flex;
  6683. flex-direction: column;
  6684. gap: 20px;
  6685. }
  6686. .pops-alert-content button{
  6687. text-wrap: wrap;
  6688. padding: 8px;
  6689. height: auto;
  6690. text-align: left;
  6691. }
  6692. `
  6693. )
  6694. });
  6695. let $content = $dialog.$shadowRoot.querySelector(
  6696. ".pops-alert-content"
  6697. );
  6698. if (discussionInfo.scriptId == null) {
  6699. (_a2 = $content.querySelector(`button[${attr_filter_key}="scriptId"]`)) == null ? void 0 : _a2.remove();
  6700. }
  6701. if (discussionInfo.scriptName == null) {
  6702. (_b = $content.querySelector(`button[${attr_filter_key}="scriptName"]`)) == null ? void 0 : _b.remove();
  6703. }
  6704. if (discussionInfo.postUserId == null) {
  6705. (_c = $content.querySelector(`button[${attr_filter_key}="postUserId"]`)) == null ? void 0 : _c.remove();
  6706. }
  6707. if (discussionInfo.replyUserId != null) {
  6708. let $replyUserIdButton = domUtils.createElement("button", {
  6709. innerHTML: i18next.t("作者id:{{text}}", {
  6710. text: discussionInfo.replyUserId
  6711. })
  6712. });
  6713. $replyUserIdButton.setAttribute(attr_filter_key, "scriptAuthorId");
  6714. $replyUserIdButton.setAttribute(
  6715. attr_filter_value,
  6716. "^" + discussionInfo.replyUserId + "$"
  6717. );
  6718. $content.appendChild($replyUserIdButton);
  6719. }
  6720. domUtils.on(
  6721. $dialog.$shadowRoot,
  6722. "click",
  6723. `button[${attr_filter_key}]`,
  6724. (event2) => {
  6725. utils.preventEvent(event2);
  6726. let $click = event2.target;
  6727. let key = $click.getAttribute(
  6728. attr_filter_key
  6729. );
  6730. let value = $click.getAttribute(attr_filter_value);
  6731. GreasyforkDiscussionsFilter.addValue(key, value);
  6732. $dialog.close();
  6733. GreasyforkDiscussionsFilter.filter();
  6734. Qmsg.success(i18next.t("添加成功"));
  6735. }
  6736. );
  6737. });
  6738. });
  6739. },
  6740. /**
  6741. * 添加【举报】按钮
  6742. */
  6743. addReportButton() {
  6744. log.info(`添加【举报】按钮`);
  6745. GreasyforkDiscussionsFilter.getElementList().forEach(($listContainer) => {
  6746. if ($listContainer.querySelector(".discussion-report-button")) {
  6747. return;
  6748. }
  6749. let $listItem = $listContainer.querySelector(
  6750. ".discussion-list-item"
  6751. );
  6752. let $meta = $listItem.querySelector(".discussion-meta");
  6753. let $ownMetaItem = domUtils.createElement(
  6754. "div",
  6755. {
  6756. className: "discussion-meta-item",
  6757. innerHTML: `
  6758. <button class="discussion-report-button" style="border-color: #ff4d4d;background-color: #ffe6e6;color: red;">${i18next.t(
  6759. "举报"
  6760. )}</button>
  6761. `
  6762. },
  6763. {
  6764. style: "flex: 0;"
  6765. }
  6766. );
  6767. let $button = $ownMetaItem.querySelector(
  6768. ".discussion-report-button"
  6769. );
  6770. $meta.appendChild($ownMetaItem);
  6771. domUtils.on($button, "click", (event) => {
  6772. utils.preventEvent(event);
  6773. const discussionInfo = GreasyforkDiscussionsFilter.parseDiscuessionListContainerInfo(
  6774. $listContainer
  6775. );
  6776. __pops.alert({
  6777. title: {
  6778. text: i18next.t("举报"),
  6779. position: "center",
  6780. html: false
  6781. },
  6782. content: {
  6783. text: (
  6784. /*html*/
  6785. `
  6786. <div class="report-item">
  6787. ${i18next.t("举报讨论:")}
  6788. <a href="${GreasyforkUrlUtils.getReportUrl(
  6789. "discussion",
  6790. discussionInfo.snippetId
  6791. )}" target="_blank">${discussionInfo.snippet}</a>
  6792. </div>
  6793. ${discussionInfo.scriptId ? (
  6794. /*html*/
  6795. `
  6796. <div class="report-item">
  6797. ${i18next.t("举报脚本:")}
  6798. <a href="${GreasyforkUrlUtils.getReportUrl(
  6799. "script",
  6800. discussionInfo.scriptId
  6801. )}" target="_blank">${discussionInfo.scriptName}</a>
  6802. </div>
  6803. `
  6804. ) : ""}
  6805. <div class="report-item">
  6806. ${i18next.t("举报用户:")}
  6807. <a href="${GreasyforkUrlUtils.getReportUrl(
  6808. "user",
  6809. discussionInfo.postUserId
  6810. )}" target="_blank">${discussionInfo.postUserName}</a>
  6811. </div>
  6812. ${discussionInfo.replyUserId && discussionInfo.replyUserId != discussionInfo.postUserId ? (
  6813. /*html*/
  6814. `
  6815. <div class="report-item">
  6816. ${i18next.t("举报用户:")}
  6817. <a href="${GreasyforkUrlUtils.getReportUrl(
  6818. "user",
  6819. discussionInfo.replyUserId
  6820. )}" target="_blank">${discussionInfo.replyUserName}</a>
  6821. </div>
  6822. `
  6823. ) : ""}
  6824. `
  6825. ),
  6826. html: true
  6827. },
  6828. btn: {
  6829. ok: {
  6830. enable: false
  6831. }
  6832. },
  6833. mask: {
  6834. enable: true,
  6835. clickEvent: {
  6836. toClose: true
  6837. }
  6838. },
  6839. drag: true,
  6840. dragLimit: true,
  6841. width: "350px",
  6842. height: "300px",
  6843. style: (
  6844. /*css*/
  6845. `
  6846. .pops-alert-content{
  6847. display: flex;
  6848. flex-direction: column;
  6849. gap: 20px;
  6850. }
  6851. .pops-alert-content .report-item{
  6852. text-wrap: wrap;
  6853. padding: 8px;
  6854. height: auto;
  6855. text-align: left;
  6856. margin: var(--button-margin-top) var(--button-margin-right)
  6857. var(--button-margin-bottom) var(--button-margin-left);
  6858. border-radius: var(--button-radius);
  6859. color: var(--button-color);
  6860. border-color: var(--button-bd-color);
  6861. background-color: var(--button-bg-color);
  6862. }
  6863. `
  6864. )
  6865. });
  6866. });
  6867. });
  6868. }
  6869. };
  6870. const GreasyforkUsers = {
  6871. init() {
  6872. PopsPanel.execMenuOnce("users-changeConsoleToTopNavigator", () => {
  6873. this.changeConsoleToTopNavigator();
  6874. });
  6875. PopsPanel.execMenuOnce("gf-scripts-filter-enable", () => {
  6876. GreasyforkScriptsFilter.init();
  6877. });
  6878. PopsPanel.execMenuOnce("beautifyCenterContent", () => {
  6879. return GreasyforkScriptsList.beautifyCenterContent();
  6880. });
  6881. },
  6882. /**
  6883. * 迁移【控制台】到顶部导航栏
  6884. */
  6885. changeConsoleToTopNavigator() {
  6886. log.info("迁移【控制台】到顶部导航栏");
  6887. CommonUtil.addBlockCSS("#about-user");
  6888. domUtils.ready(() => {
  6889. let $aboutUser = document.querySelector("#about-user");
  6890. if (!$aboutUser) {
  6891. log.error("#about-user元素不存在");
  6892. return;
  6893. }
  6894. $aboutUser = $aboutUser.cloneNode(true);
  6895. GreasyforkElementUtils.registerTopNavMenu({
  6896. name: i18next.t("控制台"),
  6897. className: "scripts-console",
  6898. clickEvent(event) {
  6899. let $dialog = __pops.alert({
  6900. title: {
  6901. text: i18next.t("控制台"),
  6902. position: "center"
  6903. },
  6904. content: {
  6905. text: "",
  6906. html: true
  6907. },
  6908. btn: {
  6909. ok: { enable: false }
  6910. },
  6911. mask: {
  6912. enable: true,
  6913. clickEvent: {
  6914. toClose: true
  6915. }
  6916. },
  6917. drag: true,
  6918. useShadowRoot: true,
  6919. width: PanelUISize.setting.width,
  6920. height: PanelUISize.setting.height,
  6921. zIndex: utils.getMaxZIndex(100),
  6922. style: (
  6923. /*css*/
  6924. `
  6925. #about-user{
  6926. border: 0;
  6927. box-shadow: none;
  6928. }
  6929. #about-user a{
  6930. color: #670000;
  6931. }
  6932. #about-user a:hover{
  6933. color: #00a3f5;
  6934. }
  6935. .text-content{
  6936. list-style-type: none;
  6937. box-shadow: rgb(221, 221, 221) 0px 0px 5px;
  6938. background-color: rgb(255, 255, 255);
  6939. box-sizing: border-box;
  6940. border-width: 1px;
  6941. border-style: solid;
  6942. border-color: rgb(187, 187, 187);
  6943. border-image: initial;
  6944. border-radius: 5px;
  6945. margin: 14px 0px;
  6946. padding: 10px 40px;
  6947. }
  6948. a.report-link{
  6949. position: absolute;
  6950. right: 0px;
  6951. font-size: smaller;
  6952. margin-right: 16px;
  6953. margin-top: 8px;
  6954. }
  6955. .notification-widget{
  6956. display: inline-block;
  6957. width: 1.2em;
  6958. height: 1.2em;
  6959. text-align: center;
  6960. line-height: 1.2em;
  6961. padding: 0;
  6962. background-color: #31708f;
  6963. border-radius: 50%;
  6964. color: #fff;
  6965. text-decoration: none;
  6966. }
  6967. `
  6968. )
  6969. });
  6970. let $content = $dialog.$shadowRoot.querySelector(
  6971. ".pops-alert-content"
  6972. );
  6973. $content.appendChild($aboutUser);
  6974. }
  6975. });
  6976. });
  6977. }
  6978. };
  6979. const beautifyContentCSS = "section.text-content {\r\n /*height: calc(100vh - 100px);*/\r\n /*overflow-y: auto;\r\n overflow-x: hidden;*/\r\n background: #eaf0ff;\r\n}\r\n\r\n.comment .user-content {\r\n border: 1px solid transparent;\r\n background: #fff;\r\n max-width: 70%;\r\n border-radius: 10px;\r\n width: fit-content;\r\n}\r\n\r\n.comment .comment-meta-spacer {\r\n flex: unset;\r\n margin-left: 15px;\r\n}\r\n.comment:not(:has(.report-link)) .comment-meta-spacer {\r\n flex: unset;\r\n margin-left: unset;\r\n margin-right: 10px;\r\n}\r\n.comment:not(:has(.report-link)) {\r\n display: flex;\r\n align-items: flex-end;\r\n flex-direction: column;\r\n}\r\n\r\n.comment:not(:has(.report-link)) .comment-meta {\r\n display: flex;\r\n flex-direction: row-reverse;\r\n}\r\n.comment:not(:has(.report-link)) .comment-meta-item {\r\n margin-left: 0px;\r\n margin-right: 15px;\r\n}";
  6980. const GreasyforkConversations = {
  6981. init() {
  6982. PopsPanel.execMenuOnce("conversations-beautifyDialogBox", () => {
  6983. return this.beautifyDialogBox();
  6984. });
  6985. domUtils.ready(() => {
  6986. PopsPanel.execMenuOnce("conversations-beautifyPrivateMessageList", () => {
  6987. this.beautifyPrivateMessageList();
  6988. });
  6989. });
  6990. },
  6991. /**
  6992. * 美化对话框
  6993. */
  6994. beautifyDialogBox() {
  6995. log.info("美化对话框");
  6996. return addStyle(beautifyContentCSS);
  6997. },
  6998. /**
  6999. * 美化私信列表
  7000. */
  7001. beautifyPrivateMessageList() {
  7002. log.info(`美化私信列表`);
  7003. addStyle(
  7004. /*css*/
  7005. `
  7006. .user-conversations-item{
  7007. list-style: none;
  7008. border: 1px solid #bfbfbf;
  7009. border-radius: 4px;
  7010. display: flex;
  7011. gap: 10px;
  7012. flex-direction: column;
  7013. padding: 4px 20px;
  7014. margin: 10px 0px;
  7015. }
  7016. .user-conversations-item .user-link-container{
  7017.  
  7018. }
  7019. .user-conversations-item .user-link-container .user-latest-send-time{
  7020. float: right;
  7021. }
  7022. .user-conversations-item .enter-coversations{
  7023. float: right;
  7024. }
  7025. `
  7026. );
  7027. document.querySelectorAll("section.text-content ul li").forEach(($li) => {
  7028. var _a2;
  7029. let $user = $li.querySelector(
  7030. 'a[href*="conversations"]'
  7031. );
  7032. let chatUrl = $user.href;
  7033. let userName = (_a2 = $user.textContent) == null ? void 0 : _a2.split(" ")[1];
  7034. let $latestMsgUser = $li.querySelector("a.user-link");
  7035. let latestSendMsgUser = null;
  7036. let latestSendMsgUserHomeUrl = null;
  7037. let latestSendMsgTimeText = null;
  7038. if ($latestMsgUser) {
  7039. latestSendMsgUser = $latestMsgUser.textContent;
  7040. latestSendMsgUserHomeUrl = $latestMsgUser.href;
  7041. let $relativeTime = $li.querySelector("relative-time");
  7042. new Date($relativeTime.getAttribute("datetime"));
  7043. latestSendMsgTimeText = $relativeTime.shadowRoot.lastChild.textContent;
  7044. }
  7045. $li.classList.add("user-conversations-item");
  7046. GreasyforkUrlUtils.getUserId();
  7047. $li.innerHTML = /*html*/
  7048. `
  7049. <div class="user-link-container">
  7050. <a class="user-link" href="${chatUrl}">${userName}</a>
  7051. <span class="user-latest-send-time">${latestSendMsgTimeText}</span>
  7052. </div>
  7053. <div class="latest-send-user-container">
  7054. ${i18next.t("最后回复:")}
  7055. <a href="${latestSendMsgUserHomeUrl}">${latestSendMsgUser}</a>
  7056. <a class="enter-coversations" href="${chatUrl}">${i18next.t("进入")}</a>
  7057. </div>
  7058. `;
  7059. });
  7060. }
  7061. };
  7062. const GreasyforkScriptsSearchElement = {
  7063. /**
  7064. * 等待脚本列表元素
  7065. */
  7066. waitScritList() {
  7067. return utils.waitNode("#browse-script-list", 1e4);
  7068. },
  7069. /**
  7070. * 添加控制区域
  7071. */
  7072. addFilterControls($scriptList) {
  7073. function getControls() {
  7074. var _a2;
  7075. let $el = document.querySelector(
  7076. "#gm-script-filter-controls"
  7077. );
  7078. return (_a2 = $el == null ? void 0 : $el.shadowRoot) == null ? void 0 : _a2.querySelector(".pops");
  7079. }
  7080. let $controls = getControls();
  7081. if ($controls) {
  7082. return $controls;
  7083. }
  7084. let $controlsContainer = domUtils.createElement("div", {
  7085. id: "gm-script-filter-controls"
  7086. });
  7087. let shadowRoot = $controlsContainer.attachShadow({ mode: "open" });
  7088. shadowRoot.appendChild(
  7089. domUtils.createElement("style", {
  7090. innerHTML: (
  7091. /*css*/
  7092. `
  7093. ${__pops.config.cssText.index}
  7094.  
  7095. ${__pops.config.cssText.common}
  7096.  
  7097. ${__pops.config.cssText.panelCSS}
  7098. `
  7099. )
  7100. })
  7101. );
  7102. shadowRoot.appendChild(
  7103. domUtils.createElement("style", {
  7104. innerHTML: (
  7105. /*css*/
  7106. `
  7107. .pops{
  7108. display: flex;
  7109. align-items: center;
  7110. flex-direction: row;
  7111. gap: 10px;
  7112. padding: 10px;
  7113. }
  7114. .pops .gm-script-control-item{
  7115. display: flex;
  7116. align-items: center;
  7117. gap: 10px;
  7118. }
  7119. .pops .pops-panel-item-left-main-text{
  7120. display: flex;
  7121. align-items: center;
  7122. margin: 0px;
  7123. padding: 0px;
  7124. }
  7125. `
  7126. )
  7127. })
  7128. );
  7129. let $pops = domUtils.createElement("div", {
  7130. className: "pops"
  7131. });
  7132. shadowRoot.appendChild($pops);
  7133. domUtils.before($scriptList, $controlsContainer);
  7134. return $pops;
  7135. }
  7136. };
  7137. const GreasyforkScriptsSearch = {
  7138. init() {
  7139. domUtils.ready(() => {
  7140. GreasyforkScriptsSearchElement.waitScritList().then(($scriptList) => {
  7141. if (!$scriptList) {
  7142. log.error("未找到脚本列表节点,无法继续执行");
  7143. return;
  7144. }
  7145. let $filterControlsContainer = GreasyforkScriptsSearchElement.addFilterControls($scriptList);
  7146. this.addFilterControlsItem($filterControlsContainer);
  7147. });
  7148. });
  7149. },
  7150. /**
  7151. * 添加过滤项
  7152. */
  7153. addFilterControlsItem($filterControlsContainer) {
  7154. let controlsConfig = [
  7155. {
  7156. name: i18next.t("名称-全词匹配"),
  7157. ENABLE_KEY: "gf-script-search-filterScriptTitleWholeWordMatching",
  7158. STORAGE_KEY: "gf-script-search-filterScriptTitleWholeWordMatching-enable",
  7159. callback: (searchText, scriptInfo) => {
  7160. return !scriptInfo.scriptName.includes(searchText);
  7161. }
  7162. },
  7163. {
  7164. name: i18next.t("描述-全词匹配"),
  7165. ENABLE_KEY: "gf-script-search-filterScriptDescWholeWordMatching",
  7166. STORAGE_KEY: "gf-script-search-filterScriptDescWholeWordMatching-enable",
  7167. callback: (searchText, scriptInfo) => {
  7168. return !scriptInfo.scriptDescription.includes(searchText);
  7169. }
  7170. },
  7171. {
  7172. name: i18next.t("作者名称-全词匹配"),
  7173. ENABLE_KEY: "gf-script-search-filterScriptAuthorNameWholeWordMatching",
  7174. STORAGE_KEY: "gf-script-search-filterScriptAuthorNameWholeWordMatching-enable",
  7175. callback: (searchText, scriptInfo) => {
  7176. return !scriptInfo.scriptAuthorName.includes(searchText);
  7177. }
  7178. }
  7179. ];
  7180. function callback() {
  7181. let searchParams = new URLSearchParams(window.location.search);
  7182. let searchText = searchParams.get("q").trim();
  7183. if (searchText == "") {
  7184. return;
  7185. }
  7186. let allScriptsList = GreasyforkScriptsFilter.getElementList();
  7187. allScriptsList.forEach(($scriptList) => {
  7188. let scriptInfo = parseScriptListInfo($scriptList);
  7189. let fitlerFlagList = controlsConfig.map((controlsConfig2) => {
  7190. let enable = _GM_getValue(controlsConfig2.STORAGE_KEY);
  7191. if (!enable) {
  7192. return;
  7193. }
  7194. return controlsConfig2.callback(searchText, scriptInfo);
  7195. }).filter((item) => typeof item === "boolean");
  7196. if (fitlerFlagList.length !== 0) {
  7197. let flag = false;
  7198. fitlerFlagList.forEach((enable) => {
  7199. flag = flag || enable;
  7200. });
  7201. if (flag) {
  7202. domUtils.hide($scriptList, false);
  7203. } else {
  7204. domUtils.show($scriptList, false);
  7205. }
  7206. } else {
  7207. domUtils.show($scriptList, false);
  7208. }
  7209. });
  7210. }
  7211. controlsConfig.forEach((controlConfig) => {
  7212. if (!PopsPanel.getValue(controlConfig.ENABLE_KEY)) {
  7213. return;
  7214. }
  7215. log.info(`添加按钮${controlConfig.name}`);
  7216. let panelHandleContentUtils = __pops.config.panelHandleContentUtils();
  7217. let $controlContainer = panelHandleContentUtils.createSectionContainerItem_switch({
  7218. type: "switch",
  7219. className: "gm-script-control-item",
  7220. text: controlConfig.name,
  7221. getValue() {
  7222. let value = _GM_getValue(controlConfig.STORAGE_KEY, false);
  7223. callback();
  7224. return value;
  7225. },
  7226. callback(event, value) {
  7227. _GM_setValue(controlConfig.STORAGE_KEY, value);
  7228. callback();
  7229. }
  7230. });
  7231. domUtils.append($filterControlsContainer, $controlContainer);
  7232. });
  7233. }
  7234. };
  7235. const Greasyfork = {
  7236. init() {
  7237. if (GreasyforkRouter.isImageSource()) {
  7238. log.info(`Router: 资源界面,不执行脚本功能`);
  7239. return;
  7240. }
  7241. PopsPanel.execMenu("checkPage", () => {
  7242. this.checkPage();
  7243. });
  7244. GreasyforkBeautify.init();
  7245. GreasyforkShortCut.init();
  7246. if (GreasyforkRouter.isScript()) {
  7247. GreasyforkScripts.init();
  7248. }
  7249. if (GreasyforkRouter.isScriptList() || GreasyforkRouter.isScriptLibraryList() || GreasyforkRouter.isScriptCodeSearch() || GreasyforkRouter.isScriptsBySite()) {
  7250. GreasyforkScriptsList.init();
  7251. }
  7252. if (GreasyforkRouter.isDiscuessions()) {
  7253. log.info(`Router: 讨论页面`);
  7254. GreasyforkForum.init();
  7255. } else if (GreasyforkRouter.isUsers()) {
  7256. log.info(`Router: 用户页面`);
  7257. GreasyforkUsers.init();
  7258. if (GreasyforkRouter.isUsersConversations()) {
  7259. log.info(`Router-next: 私聊用户页面`);
  7260. GreasyforkConversations.init();
  7261. }
  7262. } else if (GreasyforkRouter.isScriptSearch()) {
  7263. log.info(`Router: 脚本搜索页面`);
  7264. GreasyforkScriptsSearch.init();
  7265. }
  7266. PopsPanel.execMenuOnce("scripts-addOperationPanelBtnWithNavigator", () => {
  7267. this.addOperationPanelBtnWithNavigator();
  7268. });
  7269. domUtils.ready(() => {
  7270. GreasyforkMenu.initEnv();
  7271. GreasyforkAccount.init();
  7272. GreasyforkRememberFormTextArea.init();
  7273. GreasyforkMenu.handleLocalGotoCallBack();
  7274. PopsPanel.execMenuOnce("fixImageWidth", () => {
  7275. Greasyfork.fixImageWidth();
  7276. });
  7277. Greasyfork.languageSelectorLocale();
  7278. PopsPanel.execMenuOnce("optimizeImageBrowsing", () => {
  7279. Greasyfork.optimizeImageBrowsing();
  7280. });
  7281. PopsPanel.execMenuOnce("overlayBedImageClickEvent", () => {
  7282. Greasyfork.overlayBedImageClickEvent();
  7283. });
  7284. if (!GreasyforkRouter.isCodeStrict()) {
  7285. PopsPanel.execMenuOnce("addMarkdownCopyButton", () => {
  7286. Greasyfork.addMarkdownCopyButton();
  7287. });
  7288. }
  7289. });
  7290. },
  7291. /**
  7292. * 修复图片宽度显示问题
  7293. */
  7294. fixImageWidth() {
  7295. if (window.innerWidth < window.innerHeight) {
  7296. log.info("修复图片显示问题");
  7297. addStyle(
  7298. /*css*/
  7299. `
  7300. img.lum-img{
  7301. width: 100% !important;
  7302. height: 100% !important;
  7303. }
  7304. `
  7305. );
  7306. }
  7307. },
  7308. /**
  7309. * 优化图片浏览
  7310. */
  7311. optimizeImageBrowsing() {
  7312. log.info("优化图片浏览");
  7313. {
  7314. addStyle(_GM_getResourceText("ViewerCSS"));
  7315. }
  7316. addStyle(
  7317. /*css*/
  7318. `
  7319. @media (max-width: 460px) {
  7320. .lum-lightbox-image-wrapper {
  7321. display:flex;
  7322. overflow: auto;
  7323. -webkit-overflow-scrolling: touch
  7324. }
  7325. .lum-lightbox-caption {
  7326. width: 100%;
  7327. position: absolute;
  7328. bottom: 0
  7329. }
  7330. .lum-lightbox-position-helper {
  7331. margin: auto
  7332. }
  7333. .lum-lightbox-inner img {
  7334. max-width:100%;
  7335. max-height:100%;
  7336. }
  7337. }
  7338. `
  7339. );
  7340. function viewIMG(imgList = [], viewIndex = 0) {
  7341. let viewerULNodeHTML = "";
  7342. imgList.forEach((item) => {
  7343. viewerULNodeHTML += `<li><img data-src="${item}" loading="lazy"></li>`;
  7344. });
  7345. let viewerULNode = domUtils.createElement("ul", {
  7346. innerHTML: viewerULNodeHTML
  7347. });
  7348. let viewer = new Viewer(viewerULNode, {
  7349. inline: false,
  7350. url: "data-src",
  7351. zIndex: utils.getMaxZIndex() + 100,
  7352. hidden: () => {
  7353. viewer.destroy();
  7354. }
  7355. });
  7356. viewIndex = viewIndex < 0 ? 0 : viewIndex;
  7357. viewer.view(viewIndex);
  7358. viewer.zoomTo(1);
  7359. viewer.show();
  7360. }
  7361. function getImgElementSrc(element) {
  7362. return element.getAttribute("data-src") || element.getAttribute("src") || element.getAttribute("alt");
  7363. }
  7364. domUtils.on(
  7365. document,
  7366. "click",
  7367. "img",
  7368. function(event) {
  7369. var _a2;
  7370. let $img = event.target;
  7371. if (((_a2 = $img.parentElement) == null ? void 0 : _a2.localName) === "a" && $img.hasAttribute("data-screenshots")) {
  7372. return;
  7373. }
  7374. if ($img.closest(".viewer-container")) {
  7375. return;
  7376. }
  7377. if ($img.closest(".lum-lightbox-position-helper")) {
  7378. return;
  7379. }
  7380. let userContentElement = $img.closest(".user-content");
  7381. let imgList = [];
  7382. let imgIndex = 0;
  7383. let imgElementList = [];
  7384. let currentImgSrc = getImgElementSrc($img);
  7385. if (currentImgSrc) {
  7386. if (currentImgSrc.startsWith("https://img.shields.io")) {
  7387. return;
  7388. } else if (currentImgSrc.startsWith("/vite/assets/")) {
  7389. return;
  7390. }
  7391. }
  7392. if (userContentElement) {
  7393. userContentElement.querySelectorAll("img").forEach((childImgElement) => {
  7394. imgElementList.push(childImgElement);
  7395. let imgSrc = getImgElementSrc(childImgElement);
  7396. let $parent = childImgElement.parentElement;
  7397. if (($parent == null ? void 0 : $parent.localName) === "a") {
  7398. imgSrc = $parent.getAttribute("data-href") || $parent.href;
  7399. }
  7400. imgList.push(imgSrc);
  7401. });
  7402. imgIndex = imgElementList.indexOf($img);
  7403. if (imgIndex === -1) {
  7404. imgIndex = 0;
  7405. }
  7406. } else {
  7407. imgList.push(currentImgSrc);
  7408. imgIndex = 0;
  7409. }
  7410. log.success("点击浏览图片👉", imgList, imgIndex);
  7411. viewIMG(imgList, imgIndex);
  7412. }
  7413. );
  7414. document.querySelectorAll(".user-screenshots").forEach((element) => {
  7415. let linkElement = element.querySelector("a");
  7416. if (!linkElement) {
  7417. return;
  7418. }
  7419. let imgSrc = linkElement.getAttribute("data-href") || linkElement.getAttribute("href");
  7420. let imgElement = element.querySelector("img");
  7421. if (!imgElement) {
  7422. return;
  7423. }
  7424. imgElement.setAttribute("data-screenshots", "true");
  7425. imgElement.setAttribute("data-src", imgSrc);
  7426. linkElement.setAttribute("href", "javascript:;");
  7427. domUtils.after(linkElement, imgElement);
  7428. linkElement.remove();
  7429. });
  7430. },
  7431. /**
  7432. * 覆盖图床图片的parentElement的a标签
  7433. */
  7434. overlayBedImageClickEvent() {
  7435. log.info("覆盖图床图片的parentElement的a标签");
  7436. document.querySelectorAll(".user-content a>img").forEach((imgElement) => {
  7437. let $link = imgElement.parentElement;
  7438. let url = $link.getAttribute("href");
  7439. $link.setAttribute("data-href", url);
  7440. $link.removeAttribute("href");
  7441. if (url.startsWith("/rails/active_storage/blobs/redirect")) {
  7442. log.info(`该图片是上传到Greasyfork的图片,拦截默认行为,不做提示`);
  7443. return;
  7444. }
  7445. domUtils.on($link, "click", () => {
  7446. Qmsg.warning(
  7447. /*html*/
  7448. `<div style="overflow-wrap: anywhere;">${i18next.t(
  7449. "拦截跳转:"
  7450. )}<a href="${url}" target="_blank">${url}</a></div>`,
  7451. {
  7452. html: true,
  7453. zIndex: utils.getMaxZIndex() + 105
  7454. }
  7455. );
  7456. });
  7457. });
  7458. },
  7459. /**
  7460. * 在Markdown右上角添加复制按钮
  7461. */
  7462. addMarkdownCopyButton() {
  7463. log.info("在Markdown右上角添加复制按钮");
  7464. addStyle(
  7465. /*css*/
  7466. `
  7467. pre{
  7468. position: relative;
  7469. margin-bottom: 0px !important;
  7470. width: 100%;
  7471. }
  7472. `
  7473. );
  7474. addStyle(
  7475. /*css*/
  7476. `
  7477. .snippet-clipboard-content{
  7478. display: flex;
  7479. justify-content: space-between;
  7480. background: rgb(246, 248, 250);
  7481. margin-bottom: 16px;
  7482. position: relative;
  7483. }
  7484. .zeroclipboard-container {
  7485. right: 0;
  7486. top: 0;
  7487. position: absolute;
  7488. box-sizing: border-box;
  7489. display: flex;
  7490. font-size: 16px;
  7491. line-height: 24px;
  7492. text-size-adjust: 100%;
  7493. overflow-wrap: break-word;
  7494. width: fit-content;
  7495. height: fit-content;
  7496. }
  7497. .zeroclipboard-container svg{
  7498. vertical-align: text-bottom;
  7499. display: inline-block;
  7500. overflow: visible;
  7501. fill: currentColor;
  7502. margin: 8px;
  7503. }
  7504. .zeroclipboard-container svg[aria-hidden="true"]{
  7505. display: none;
  7506. }
  7507. clipboard-copy.js-clipboard-copy {
  7508. position: relative;
  7509. padding: 0px;
  7510. color: rgb(36, 41, 47);
  7511. background-color: rgb(246, 248, 250);
  7512. transition: 80ms cubic-bezier(0.33, 1, 0.68, 1);
  7513. transition-property: color,background-color,box-shadow,border-color;
  7514. display: inline-block;
  7515. font-size: 14px;
  7516. line-height: 20px;
  7517. white-space: nowrap;
  7518. vertical-align: middle;
  7519. cursor: pointer;
  7520. -webkit-user-select: none;
  7521. user-select: none;
  7522. border: 1px solid rgba(31, 35, 40, 0.15);
  7523. -webkit-appearance: none;
  7524. appearance: none;
  7525. box-shadow: rgba(31, 35, 40, 0.04) 0px 1px 0px 0px, rgba(255, 255, 255, 0.25) 0px 1px 0px 0px inset;
  7526. margin: 8px;
  7527. overflow-wrap: break-word;
  7528. text-wrap: nowrap;
  7529. border-radius: 6px;
  7530. }
  7531. clipboard-copy.js-clipboard-copy[success]{
  7532. border-color: rgb(31, 136, 61);
  7533. box-shadow: 0 0 0 0.2em rgba(52,208,88,.4);
  7534. }
  7535. clipboard-copy.js-clipboard-copy:hover{
  7536. background-color: rgb(243, 244, 246);
  7537. border-color: rgba(31, 35, 40, 0.15);
  7538. transition-duration: .1s;
  7539. }
  7540. clipboard-copy.js-clipboard-copy:active{
  7541. background-color: rgb(235, 236, 240);
  7542. border-color: rgba(31, 35, 40, 0.15);
  7543. transition: none;
  7544. }
  7545. `
  7546. );
  7547. addStyle(
  7548. /*css*/
  7549. `
  7550. .pops-tip.github-tooltip {
  7551. border-radius: 6px;
  7552. padding: 6px 8px;
  7553. }
  7554. .pops-tip.github-tooltip, .pops-tip.github-tooltip .pops-tip-arrow::after {
  7555. background: rgb(36, 41, 47);
  7556. color: #fff;
  7557. }
  7558. .pops-tip.github-tooltip .pops-tip-arrow::after {
  7559. width: 8px;
  7560. height: 8px;
  7561. }
  7562. `
  7563. );
  7564. function createCopyElement() {
  7565. let $copy = domUtils.createElement("div", {
  7566. className: "zeroclipboard-container",
  7567. innerHTML: (
  7568. /*html*/
  7569. `
  7570. <clipboard-copy class="js-clipboard-copy">
  7571. <svg height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon-copy">
  7572. <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>
  7573. </svg>
  7574. <svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" class="octicon-check-copy">
  7575. <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>
  7576. </svg>
  7577. </clipboard-copy>
  7578. `
  7579. )
  7580. });
  7581. let clipboardCopyElement = $copy.querySelector(
  7582. ".js-clipboard-copy"
  7583. );
  7584. let octiconCopyElement = $copy.querySelector(
  7585. ".octicon-copy"
  7586. );
  7587. let octiconCheckCopyElement = $copy.querySelector(
  7588. ".octicon-check-copy"
  7589. );
  7590. domUtils.on($copy, "click", () => {
  7591. let $parent = domUtils.parent($copy);
  7592. let $code = $parent.querySelector("code");
  7593. if (!$code && $parent.className.includes("prettyprinted")) {
  7594. $code = $copy.parentElement;
  7595. }
  7596. if (!$code) {
  7597. $code = $parent.querySelector("pre");
  7598. }
  7599. if (!$code) {
  7600. Qmsg.error(i18next.t("未找到{{selector}}元素", { selector: "code" }));
  7601. return;
  7602. }
  7603. utils.setClip(domUtils.text($code));
  7604. clipboardCopyElement.setAttribute("success", "true");
  7605. octiconCopyElement.setAttribute("aria-hidden", "true");
  7606. octiconCheckCopyElement.removeAttribute("aria-hidden");
  7607. let tooltip = __pops.tooltip({
  7608. target: clipboardCopyElement,
  7609. content: i18next.t("✅ 复制成功!"),
  7610. position: "left",
  7611. className: "github-tooltip",
  7612. alwaysShow: true
  7613. });
  7614. tooltip.toolTip.onToolTipAnimationFinishEvent();
  7615. setTimeout(() => {
  7616. clipboardCopyElement.removeAttribute("success");
  7617. octiconCheckCopyElement.setAttribute("aria-hidden", "true");
  7618. octiconCopyElement.removeAttribute("aria-hidden");
  7619. tooltip.toolTip.close();
  7620. }, 2e3);
  7621. });
  7622. return $copy;
  7623. }
  7624. $$("pre").forEach((preElement) => {
  7625. let zeroclipboardElement = preElement.querySelector(
  7626. "div.zeroclipboard-container"
  7627. );
  7628. if (zeroclipboardElement) {
  7629. return;
  7630. }
  7631. let copyElement = createCopyElement();
  7632. let snippetClipboardContentElement = domUtils.createElement("div", {
  7633. className: "snippet-clipboard-content"
  7634. });
  7635. domUtils.before(preElement, snippetClipboardContentElement);
  7636. snippetClipboardContentElement.appendChild(preElement);
  7637. snippetClipboardContentElement.appendChild(copyElement);
  7638. });
  7639. },
  7640. /**
  7641. * 固定当前语言
  7642. */
  7643. languageSelectorLocale() {
  7644. let localeLanguage = PopsPanel.getValue("language-selector-locale");
  7645. let currentLocaleLanguage = window.location.pathname.split("/").filter((item) => Boolean(item))[0];
  7646. log.success("选择语言:" + localeLanguage);
  7647. log.success("当前语言:" + currentLocaleLanguage);
  7648. if (utils.isNull(localeLanguage)) {
  7649. return;
  7650. }
  7651. if (localeLanguage === currentLocaleLanguage) {
  7652. return;
  7653. } else {
  7654. let timer = null;
  7655. let url = GreasyforkUrlUtils.getSwitchLanguageUrl(localeLanguage);
  7656. GreasyforkApi.switchLanguage(url);
  7657. log.success("新Url:" + url);
  7658. Qmsg.loading(
  7659. i18next.t(
  7660. "当前语言:{{currentLocaleLanguage}},,3秒后切换至:{{localeLanguage}}",
  7661. {
  7662. currentLocaleLanguage,
  7663. localeLanguage
  7664. }
  7665. ),
  7666. {
  7667. timeout: 3e3,
  7668. showClose: true,
  7669. onClose() {
  7670. clearTimeout(timer);
  7671. }
  7672. }
  7673. );
  7674. Qmsg.info(i18next.t("导航至:") + url, {
  7675. timeout: 3e3
  7676. });
  7677. timer = setTimeout(() => {
  7678. window.location.href = url;
  7679. }, 3e3);
  7680. }
  7681. },
  7682. /**
  7683. * 检测gf页面是否正确加载,有时候会出现
  7684. * We're down for maintenance. Check back again soon.
  7685. */
  7686. checkPage() {
  7687. log.info("检测gf页面是否正确加载,有时候会出现");
  7688. domUtils.ready(() => {
  7689. if (document.body.firstElementChild && document.body.firstElementChild.localName === "p" && document.body.firstElementChild.innerText.includes(
  7690. "We're down for maintenance. Check back again soon."
  7691. )) {
  7692. let latestRefreshPageTime = parseInt(
  7693. _GM_getValue(
  7694. "greasyfork-check-page-time",
  7695. 0
  7696. )
  7697. );
  7698. let checkPageTimeout = PopsPanel.getValue(
  7699. "greasyfork-check-page-timeout",
  7700. 5
  7701. );
  7702. let checkPageTimeoutStamp = checkPageTimeout * 1e3;
  7703. if (latestRefreshPageTime && Date.now() - latestRefreshPageTime < checkPageTimeoutStamp) {
  7704. Qmsg.warning(
  7705. i18next.t("上次重载时间 {{time}},{{timeout}}秒内拒绝反复重载", {
  7706. time: utils.formatTime(
  7707. latestRefreshPageTime,
  7708. "yyyy-MM-dd HH:mm:ss"
  7709. ),
  7710. timeout: checkPageTimeout
  7711. })
  7712. );
  7713. return;
  7714. }
  7715. _GM_setValue("greasyfork-check-page-time", Date.now());
  7716. window.location.reload();
  7717. }
  7718. });
  7719. },
  7720. /**
  7721. * 在顶部导航栏添加【操作面板】按钮
  7722. */
  7723. addOperationPanelBtnWithNavigator() {
  7724. log.info("添加【操作面板】按钮");
  7725. CommonUtil.addBlockCSS(
  7726. ".sidebarred .sidebar",
  7727. ".sidebarred-main-content .open-sidebar"
  7728. );
  7729. addStyle(
  7730. /*css*/
  7731. `
  7732. .sidebarred .sidebarred-main-content{
  7733. max-width: 100%;
  7734. }
  7735. `
  7736. );
  7737. domUtils.ready(() => {
  7738. document.querySelector("#site-nav nav");
  7739. document.querySelector(
  7740. "#site-nav .with-submenu nav"
  7741. );
  7742. let $scriptsOptionGroups = document.querySelector("#script-list-option-groups") || document.querySelector(".list-option-groups");
  7743. if (!$scriptsOptionGroups) {
  7744. log.warn("不存在右侧面板元素#script-list-option-groups");
  7745. return;
  7746. }
  7747. $scriptsOptionGroups = $scriptsOptionGroups.cloneNode(
  7748. true
  7749. );
  7750. $scriptsOptionGroups.classList.add("option-panel-groups");
  7751. GreasyforkElementUtils.registerTopNavMenu({
  7752. name: i18next.t("操作面板"),
  7753. className: "filter-scripts",
  7754. clickEvent(event) {
  7755. let $dialog = __pops.alert({
  7756. title: {
  7757. text: i18next.t("操作面板"),
  7758. position: "center"
  7759. },
  7760. content: {
  7761. text: "",
  7762. html: true
  7763. },
  7764. btn: {
  7765. ok: { enable: false }
  7766. },
  7767. mask: {
  7768. enable: true,
  7769. clickEvent: {
  7770. toClose: true
  7771. }
  7772. },
  7773. drag: true,
  7774. useShadowRoot: true,
  7775. width: PanelUISize.setting.width,
  7776. height: PanelUISize.setting.height,
  7777. zIndex: utils.getMaxZIndex(100),
  7778. style: (
  7779. /*css*/
  7780. `
  7781. .pops-drawer-content div:first-child{
  7782. margin: 20px;
  7783. }
  7784. .option-panel-groups > div{
  7785. }
  7786. .option-panel-groups ul{
  7787. margin: .5em 0 0;
  7788. list-style-type: none;
  7789. padding: 1em 0;
  7790. box-shadow: 0 0 5px #ddd;
  7791. border: 1px solid #BBBBBB;
  7792. border-radius: 5px;
  7793. background-color: #fff;
  7794. }
  7795. .option-panel-groups ul li{
  7796. }
  7797. li.list-current{
  7798. border-left: 7px solid #800;
  7799. box-shadow: inset 0 1px #0000001a, inset 0 -1px #0000001a;
  7800. margin: 0 0 0 -4px;
  7801. padding: .4em 1em .4em calc(1em - 3px);
  7802. background: linear-gradient(#fff, #eee);
  7803. }
  7804. .list-option-group a {
  7805. padding: .35em 1em;
  7806. display: block;
  7807. }
  7808. .list-option-group {
  7809. margin-bottom: 1em;
  7810. }
  7811. form.sidebar-search{
  7812. display: flex;
  7813. align-items: center;
  7814. gap: 10px;
  7815. }
  7816. form.sidebar-search input[type="search"]{
  7817. display: inline-flex;
  7818. justify-content: center;
  7819. align-items: center;
  7820. line-height: 1;
  7821. height: 32px;
  7822. white-space: nowrap;
  7823. cursor: text;
  7824. text-align: center;
  7825. box-sizing: border-box;
  7826. outline: 0;
  7827. transition: 0.1s;
  7828. font-weight: 500;
  7829. user-select: none;
  7830. -webkit-user-select: none;
  7831. -moz-user-select: none;
  7832. -ms-user-select: none;
  7833. vertical-align: middle;
  7834. -webkit-appearance: none;
  7835. appearance: none;
  7836. background-color: transparent;
  7837. border: 0;
  7838. padding: 8px 8px;
  7839. font-size: 14px;
  7840. text-align: start;
  7841. /* width: 100%; */
  7842. // flex: 1;
  7843. display: flex;
  7844. align-items: center;
  7845. border: 1px solid #dcdfe6;
  7846. border-radius: 4px;
  7847. background-color: #ffffff;
  7848. }
  7849. form.sidebar-search input[type="submit"]{
  7850. width: 32px;
  7851. height: 32px;
  7852. }
  7853. `
  7854. )
  7855. });
  7856. let $content = $dialog.$shadowRoot.querySelector(
  7857. ".pops-alert-content"
  7858. );
  7859. $content.appendChild($scriptsOptionGroups);
  7860. }
  7861. });
  7862. });
  7863. }
  7864. };
  7865. PopsPanel.init();
  7866. Greasyfork.init();
  7867.  
  7868. })(Qmsg, DOMUtils, Utils, i18next, pops, Viewer);