Greasy Fork镜像 增强

增进 Greasyfork 浏览体验。

  1. // ==UserScript==
  2. // @name Greasy Fork镜像 Enhance
  3. // @name:zh-CN Greasy Fork镜像 增强
  4. // @namespace http://tampermonkey.net/
  5. // @version 0.9.2
  6. // @description Enhance your experience at Greasyfork.
  7. // @description:zh-CN 增进 Greasyfork 浏览体验。
  8. // @match https://gf.qytechs.cn/*
  9. // @author PRO
  10. // @grant GM_setValue
  11. // @grant GM_getValue
  12. // @grant GM_deleteValue
  13. // @grant GM_registerMenuCommand
  14. // @grant GM_unregisterMenuCommand
  15. // @grant GM_addValueChangeListener
  16. // @require https://github.com/PRO-2684/GM_config/releases/download/v1.2.1/config.min.js#md5=525526b8f0b6b8606cedf08c651163c2
  17. // @icon https://raw.githubusercontent.com/greasyfork-org/greasyfork/main/public/images/blacklogo16.png
  18. // @icon64 https://raw.githubusercontent.com/greasyfork-org/greasyfork/main/public/images/blacklogo96.png
  19. // @license gpl-3.0
  20. // ==/UserScript==
  21.  
  22. (function () {
  23. 'use strict';
  24. // Judge if the script should run
  25. const { contentType } = document;
  26. if (contentType !== "text/html") return;
  27.  
  28. const idPrefix = "greasyfork-enhance-";
  29. const name = GM_info.script.name;
  30.  
  31. // Config
  32. const configDesc = {
  33. $default: {
  34. autoClose: false,
  35. },
  36. filterAndSearch: {
  37. name: "🔎 Filter and Search",
  38. type: "folder",
  39. items: {
  40. anchor: {
  41. name: "*Anchor",
  42. title: "Show anchor for each heading",
  43. type: "bool",
  44. value: true,
  45. },
  46. outline: {
  47. name: "*Outline",
  48. title: "Show an outline for the page, if your screen is wide enough",
  49. type: "bool",
  50. value: true,
  51. },
  52. shortcut: {
  53. name: "Shortcut",
  54. title: "Enable keyboard shortcuts",
  55. type: "bool",
  56. value: true,
  57. },
  58. regexFilter: {
  59. name: "Regex filter",
  60. title: "Use regex to filter out matching scripts",
  61. value: "",
  62. },
  63. searchSyntax: {
  64. name: "*Search syntax",
  65. title: "Enable partial search syntax for Greasy Fork镜像 search bar",
  66. type: "bool",
  67. value: true,
  68. },
  69. },
  70. },
  71. codeblocks: {
  72. name: "📝 Code blocks",
  73. type: "folder",
  74. items: {
  75. toolbar: {
  76. name: "*Toolbar",
  77. title: "Show toolbar for code blocks, which allows copying and toggling code",
  78. type: "bool",
  79. value: true,
  80. },
  81. autoHideCode: {
  82. name: "Auto hide code",
  83. title: "Hide long code blocks by default",
  84. type: "bool",
  85. value: true,
  86. },
  87. autoHideRows: {
  88. name: "Min rows to hide",
  89. title: "Minimum number of rows to hide",
  90. type: "int",
  91. min: 1,
  92. value: 10,
  93. },
  94. tabSize: {
  95. name: "Tab size",
  96. title: "Set Tab indentation size",
  97. type: "int",
  98. min: 0,
  99. value: 4,
  100. },
  101. animation: {
  102. name: "Animation",
  103. title: "Enable animation for toggling code blocks",
  104. type: "bool",
  105. value: true,
  106. },
  107. metadata: {
  108. name: "Metadata",
  109. title: "Parses certain script metadata and displays it on the script code page",
  110. type: "bool",
  111. value: false,
  112. },
  113. }
  114. },
  115. display: {
  116. name: "🎨 Display",
  117. type: "folder",
  118. items: {
  119. hideButtons: {
  120. name: "Hide buttons",
  121. title: "Hide floating buttons added by this script",
  122. type: "bool",
  123. value: false,
  124. },
  125. stickyPagination: {
  126. name: "Sticky pagination",
  127. title: "Make pagination bar sticky",
  128. type: "bool",
  129. value: true,
  130. },
  131. flatLayout: {
  132. name: "Flat layout",
  133. title: "Use flat layout for script list and descriptions",
  134. type: "bool",
  135. value: false,
  136. },
  137. showVersion: {
  138. name: "Show version",
  139. title: "Show version number in script list",
  140. type: "bool",
  141. value: false,
  142. },
  143. navigationBar: {
  144. name: "Navigation bar",
  145. title: "Override navigation bar style",
  146. type: "enum",
  147. options: ["Default", "Desktop", "Mobile"],
  148. value: 0,
  149. },
  150. alwaysShowNotification: {
  151. name: "Always show notification",
  152. title: "Always show the notification widget",
  153. type: "bool",
  154. value: false,
  155. },
  156. },
  157. },
  158. credentials: {
  159. name: "🔑 Credentials",
  160. type: "folder",
  161. items: {
  162. autoLogin: {
  163. name: "*Auto login",
  164. title: "Automatically login to Greasy Fork镜像, if not already (only support email/password login)",
  165. type: "enum",
  166. options: ["Never", "HomepageOnly", "Always"],
  167. },
  168. captureCredentials: {
  169. name: "Capture credentials",
  170. title: "Automatically save email and password after login attempt, overwriting existing values",
  171. type: "bool",
  172. value: false,
  173. },
  174. email: {
  175. name: "Email",
  176. title: "Email address for auto login",
  177. type: "text",
  178. value: "",
  179. },
  180. password: {
  181. name: "Password",
  182. title: "Password for auto login",
  183. type: "password",
  184. value: "",
  185. formatter: (prop, value, desc) => `${desc.name}: ${value ? "*".repeat(value.length) : ""}`,
  186. },
  187. },
  188. },
  189. other: {
  190. name: "🔧 Other",
  191. type: "folder",
  192. items: {
  193. shortLink: {
  194. name: "Short link",
  195. title: "Display a shortened link to current script",
  196. type: "bool",
  197. value: true,
  198. },
  199. libAlternativeUrl: {
  200. name: "Alternative URLs for library",
  201. title: "Show a list of alternative URLs for a given library",
  202. type: "bool",
  203. value: false,
  204. },
  205. imageProxy: {
  206. name: "*Image proxy",
  207. title: "Use `wsrv.nl` as proxy for user-uploaded images",
  208. type: "bool",
  209. value: false,
  210. },
  211. lazyImage: {
  212. name: "*Lazy image",
  213. title: "Load user images lazily",
  214. type: "bool",
  215. value: false,
  216. },
  217. debug: {
  218. name: "Debug",
  219. title: "Enable debug mode",
  220. type: "bool",
  221. value: false,
  222. },
  223. }
  224. }
  225. };
  226. const config = new GM_config(configDesc);
  227. // CSS
  228. /**
  229. * Dynamic styles for the bool type.
  230. * @type {Object<string, string>}
  231. */
  232. const dynamicStyles = {
  233. "codeblocks.animation": `
  234. /* Toggle code animation */
  235. pre > code { transition: height 0.5s ease-in-out 0s; }
  236. /* Adapted from animate.css - https://animate.style/ */
  237. :root { --animate-duration: 1s; --animate-delay: 1s; --animate-repeat: 1; }
  238. .animate__animated { animation-duration: var(--animate-duration); animation-fill-mode: both; }
  239. .animate__animated.animate__fastest { animation-duration: calc(var(--animate-duration) / 3); }
  240. @keyframes tada {
  241. from { transform: scale3d(1, 1, 1); }
  242. 10%, 20% { transform: scale3d(0.9, 0.9, 0.9) rotate3d(0, 0, 1, -3deg); }
  243. 30%, 50%, 70%, 90% { transform: scale3d(1.1, 1.1, 1.1) rotate3d(0, 0, 1, 3deg); }
  244. 40%, 60%, 80% { transform: scale3d(1.1, 1.1, 1.1) rotate3d(0, 0, 1, -3deg); }
  245. to { transform: scale3d(1, 1, 1); }
  246. }
  247. .animate__tada { animation-name: tada; }
  248. @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } }
  249. .animate__fadeIn { animation-name: fadeIn; }
  250. @keyframes fadeOut { from { opacity: 1; } to { opacity: 0; } }
  251. .animate__fadeOut { -webkit-animation-name: fadeOut; animation-name: fadeOut; }
  252. `,
  253. "display.hideButtons": `div#float-buttons { display: none; }`,
  254. "display.stickyPagination": `.sidebarred-main-content > .pagination { position: sticky; bottom: 0; backdrop-filter: blur(5px); padding: 0.5em; }`,
  255. "display.flatLayout": `
  256. .script-list > li {
  257. &:not(.ad-entry) { padding-right: 0; }
  258. article {
  259. display: flex; flex-direction: row; justify-content: space-between; align-items: center;
  260. > .script-meta-block {
  261. width: 40%; column-gap: 0;
  262. > .inline-script-stats {
  263. margin: 0;
  264. > dd { white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
  265. }
  266. }
  267. > h2 {
  268. width: 60%; overflow: hidden; text-overflow: ellipsis; margin-right: 0.5em; padding-right: 0.5em; border-right: 1px solid #88888888;
  269. > .script-link { white-space: nowrap; }
  270. > .script-description { display: block; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
  271. }
  272. }
  273. &[data-script-type="library"] > article {
  274. > h2 { width: 80%; }
  275. > .script-meta-block { width: 20%; column-count: 1; }
  276. }
  277. }
  278. @media (max-width: 600px) {
  279. .script-list > li {
  280. &[data-script-type="library"] > article > div.script-meta-block { width: 40%; }
  281. &:not([data-script-type="library"]) > article {
  282. display: block;
  283. > h2 { width: unset; border-right: none; }
  284. > .script-meta-block { column-count: 2; }
  285. }
  286. > article > div.script-meta-block { width: unset; column-gap: 0; }
  287. }
  288. }
  289. .showing-all-languages .badge-js, .showing-all-languages .badge-css, .script-type { display: none; }
  290. #script-info .script-meta-block { float: right; column-count: 1; max-width: 300px; border-left: 1px solid #DDDDDD; margin-left: 1em; padding-left: 1em; }
  291. #additional-info { width: calc(100% - 2em - 2px); }
  292. `,
  293. "display.showVersion": `.script-list > li[data-script-version]::before { content: "@" attr(data-script-version); position: absolute; translate: 0 -1em; color: grey; font-size: smaller; }`,
  294. };
  295. /**
  296. * Dynamic styles for the enum type.
  297. * @type {Object<string, Array<string>>}
  298. */
  299. const enumStyles = {
  300. "display.navigationBar": [
  301. "/* Default */",
  302. "/* Desktop */ #main-header { #site-nav { display: block; } #mobile-nav { display: none; } }",
  303. "/* Mobile */ #main-header { #site-nav { display: none; } #mobile-nav { display: block; } }",
  304. ]
  305. };
  306. // Common Helper Functions
  307. const $ = document.querySelector.bind(document);
  308. const $$ = document.querySelectorAll.bind(document);
  309. const body = $("body");
  310. function log(...args) {
  311. if (config.get("other.debug")) {
  312. console.log(`[${name}]`, ...args);
  313. }
  314. }
  315. function injectCSS(id, css) {
  316. const style = document.head.appendChild(document.createElement("style"));
  317. style.id = idPrefix + id;
  318. style.textContent = css;
  319. return style;
  320. }
  321. function cssHelper(id, enable) {
  322. const current = document.getElementById(idPrefix + id);
  323. if (current) {
  324. current.disabled = !enable;
  325. } else if (enable) {
  326. injectCSS(id, dynamicStyles[id]);
  327. }
  328. }
  329. /**
  330. * Helper function to configure enum styles.
  331. * @param {string} id The ID of the style.
  332. * @param {string} mode The mode to set.
  333. */
  334. function enumStyleHelper(id, mode) {
  335. const style = document.getElementById(idPrefix + id) ?? injectCSS(id, "");
  336. style.textContent = enumStyles[id][mode];
  337. }
  338. // Basic css
  339. injectCSS("basic", `
  340. html { scroll-behavior: smooth; }
  341. a.anchor::before { content: "#"; }
  342. a.anchor { opacity: 0; text-decoration: none; padding: 0px 0.5em; transition: all 0.25s ease-in-out; }
  343. h1:hover>a.anchor, h2:hover>a.anchor, h3:hover>a.anchor,
  344. h4:hover>a.anchor, h5:hover>a.anchor, h6:hover>a.anchor { opacity: 1; transition: all 0.25s ease-in-out; }
  345. a.button { margin: 0.5em 0 0 0; display: flex; align-items: center; justify-content: center; text-decoration: none; color: black; background-color: #a42121ab; border-radius: 50%; width: 2em; height: 2em; font-size: 1.8em; font-weight: bold; }
  346. div.code-toolbar { display: flex; gap: 1em; }
  347. a.code-operation { cursor: pointer; font-style: italic; }
  348. div.lum-lightbox { z-index: 2; }
  349. #float-buttons { position: fixed; bottom: 1em; right: 1em; display: flex; flex-direction: column; user-select: none; z-index: 1; }
  350. aside.panel { display: none; }
  351. .dynamic-opacity { transition: opacity 0.2s ease-in-out; opacity: 0.2; }
  352. .dynamic-opacity:hover { opacity: 0.8; }
  353. input[type=file] { border-style: dashed; border-radius: 0.5em; border-color: gray; padding: 0.5em; background: rgba(169, 169, 169, 0.4); transition-property: border-color, background; transition-duration: 0.25s; transition-timing-function: ease-in-out; }
  354. input[type=file]:hover { border-color: black; background: rgba(169, 169, 169, 0.6); }
  355. input[type=file]::file-selector-button { border: 1px solid; border-radius: 0.3em; transition: background 0.25s ease-in-out; background: rgba(169, 169, 169, 0.7); }
  356. input[type=file]::file-selector-button:hover { background: rgba(169, 169, 169, 1); }
  357. table { border: 1px solid #8d8d8d; border-collapse: collapse; width: auto; }
  358. table td, table th { padding: 0.5em 0.75em; vertical-align: middle; border: 1px solid #8d8d8d; }
  359. @media (any-hover: none) { .dynamic-opacity { opacity: 0.8; } .dynamic-opacity:hover { opacity: 0.8; } }
  360. @media screen and (min-width: 767px) {
  361. aside.panel { display: contents; line-height: 1.5; }
  362. ul.outline { position: sticky; float: right; padding: 0 0 0 0.5em; margin: 0 0.5em -99vh; max-height: 80vh; border: 1px solid #BBBBBB; border-left: 2px solid #F2E5E5; box-shadow: 0 0 5px #ddd; background: linear-gradient(to right, #fcf1f1, #FFF 1em); list-style: none; width: 10.5%; color: gray; border-radius: 5px; overflow-y: scroll; z-index: 1; }
  363. ul.outline > li { overflow: hidden; text-overflow: ellipsis; }
  364. ul.outline > li > a { color: gray; white-space: nowrap; text-decoration: none; }
  365. }
  366. pre > code { overflow: hidden; display: block; }
  367. ul { padding-left: 1.5em; }
  368. .script-list > .regex-filtered { display: none; }
  369. #greasyfork-enhance-regex-filter-tip { float: right; color: grey; }
  370. @media screen and (max-width: 800px) { #greasyfork-enhance-regex-filter-tip { display: none; } }`);
  371.  
  372. // Buttons
  373. const buttons = body.appendChild(document.createElement("div"));
  374. buttons.id = "float-buttons";
  375. const goToTop = buttons.appendChild(document.createElement("a"));
  376. goToTop.classList.add("button");
  377. goToTop.classList.add("dynamic-opacity");
  378. goToTop.href = "#top";
  379. goToTop.text = "↑";
  380. // Double click to get to top
  381. body.addEventListener("dblclick", (e) => {
  382. if (e.target === body) {
  383. goToTop.click();
  384. }
  385. });
  386. // Fix current tab link
  387. const tab = $("ul#script-links > li.current");
  388. if (tab) {
  389. const link = tab.appendChild(document.createElement("a"));
  390. link.href = window.location.pathname;
  391. link.appendChild(tab.firstChild);
  392. }
  393. const parts = window.location.pathname.split("/");
  394. if (parts.length <= 2 || (parts.length == 3 && parts[2] === '')) {
  395. const banner = $("header#main-header div#site-name");
  396. const img = banner.querySelector("img");
  397. const text = banner.querySelector("#site-name-text > h1");
  398. const link1 = document.createElement("a");
  399. link1.href = window.location.pathname;
  400. img.parentNode.replaceChild(link1, img);
  401. link1.appendChild(img);
  402. const link2 = document.createElement("a");
  403. link2.href = window.location.pathname;
  404. link2.textContent = text.textContent;
  405. text.textContent = "";
  406. text.appendChild(link2);
  407. }
  408.  
  409. // Filter and Search
  410. // Anchor & Outline
  411. if (config.get("filterAndSearch.anchor") || config.get("filterAndSearch.outline")) {
  412. function sanitify(s) {
  413. // Remove emojis (such a headache)
  414. s = s.replaceAll(/([\uE000-\uF8FF]|\uD83C[\uDC00-\uDFFF]|\uD83D[\uDC00-\uDFFF]|[\u2580-\u27BF]|\uD83E[\uDD10-\uDEFF]|\uFE0F)/g, "");
  415. // Trim spaces and newlines
  416. s = s.trim();
  417. // Replace spaces
  418. s = s.replaceAll(" ", "-");
  419. s = s.replaceAll("%20", "-");
  420. // No more multiple "-"
  421. s = s.replaceAll(/-+/g, "-");
  422. return s;
  423. }
  424. function process(outline, node) { // Add anchor and assign id to given node; Add to outline. Return true if node is actually processed.
  425. if (node.childElementCount > 1 || node.classList.length > 0) return false; // Ignore complex nodes
  426. const text = node.textContent;
  427. if (!node.id) { // If the node has no id
  428. node.id = sanitify(text); // Then assign id
  429. }
  430. // Add anchors
  431. if (config.get("filterAndSearch.anchor")) {
  432. const anchor = node.appendChild(document.createElement('a'));
  433. anchor.className = 'anchor';
  434. anchor.href = '#' + node.id;
  435. }
  436. if (outline) {
  437. const link = outline.appendChild(document.createElement("li"))
  438. .appendChild(document.createElement("a"));
  439. link.href = "#" + node.id;
  440. link.text = text;
  441. }
  442. return true;
  443. }
  444.  
  445. // Outline & Anchors
  446. const isScript = /^\/[^\/]+\/scripts/;
  447. const isSpecificScript = /^\/[^\/]+\/scripts\/\d+/;
  448. const isDisccussion = /^\/[^\/]+\/discussions/;
  449. const path = window.location.pathname;
  450. if ((!isScript.test(path) && !isDisccussion.test(path)) || isSpecificScript.test(path)) {
  451. let panel = null, outline = null;
  452. if (config.get("filterAndSearch.outline")) {
  453. panel = body.insertBefore(document.createElement("aside"), $("body > div.width-constraint"));
  454. panel.className = "panel";
  455. const referenceNode = $("body > div.width-constraint > section");
  456. outline = panel.appendChild(document.createElement("ul"));
  457. outline.classList.add("outline");
  458. outline.classList.add("dynamic-opacity");
  459. outline.style.top = referenceNode ? getComputedStyle(referenceNode).marginTop : "1em";
  460. outline.style.marginTop = outline.style.top;
  461. }
  462. let flag = false;
  463. $$("body > div.width-constraint h1, h2, h3, h4, h5, h6").forEach((node) => {
  464. flag = process(outline, node) || flag; // Not `flag || process(node)`!
  465. });
  466. if (!flag) {
  467. panel?.remove();
  468. }
  469. }
  470. // Navigate to hash
  471. const hash = window.location.hash.slice(1);
  472. if (hash) {
  473. const ele = document.getElementById(decodeURIComponent(hash));
  474. if (ele) {
  475. ele.scrollIntoView();
  476. }
  477. }
  478. }
  479. // Shortcut
  480. function submitOnCtrlEnter(e) {
  481. const form = this.form;
  482. if (!form) return;
  483. // Ctrl + Enter to submit
  484. if (e.ctrlKey && e.key === "Enter") {
  485. form.submit();
  486. }
  487. }
  488. function handleShortcut(e) {
  489. const ele = document.activeElement;
  490. // Ignore key combinations
  491. if (e.ctrlKey || e.altKey || e.metaKey || e.shiftKey) {
  492. return;
  493. }
  494. // Do not interfere with input elements
  495. if (ele.tagName === "INPUT" || ele.tagName === "TEXTAREA" || ele.getAttribute("contenteditable") === "true") {
  496. if (e.key === "Escape") {
  497. e.preventDefault();
  498. ele.blur(); // Escape to blur
  499. }
  500. return;
  501. }
  502. // Do not interfere with input methods
  503. if (e.isComposing || e.keyCode === 229) {
  504. return;
  505. }
  506. // Focus on search bar
  507. switch (e.key) {
  508. case "Enter": {
  509. const input = $("input[type=search]") || $("input[type=text]") || $("textarea");
  510. if (input) {
  511. e.preventDefault();
  512. input.focus();
  513. }
  514. break;
  515. }
  516. case "ArrowLeft":
  517. $("a.previous_page")?.click();
  518. break;
  519. case "ArrowRight":
  520. $("a.next_page")?.click();
  521. break;
  522. }
  523. }
  524. let shortcutEnabled = false;
  525. function shortcut(enable) {
  526. const textAreas = $$("textarea");
  527. if (!shortcutEnabled && enable) {
  528. for (const textarea of textAreas) {
  529. textarea.addEventListener("keyup", submitOnCtrlEnter);
  530. }
  531. document.addEventListener("keydown", handleShortcut);
  532. shortcutEnabled = true;
  533. } else if (shortcutEnabled && !enable) {
  534. for (const textarea of textAreas) {
  535. textarea.removeEventListener("keyup", submitOnCtrlEnter);
  536. }
  537. document.removeEventListener("keydown", handleShortcut);
  538. shortcutEnabled = false;
  539. }
  540. }
  541. shortcut(config.get("filterAndSearch.shortcut"));
  542. // Regex filter
  543. const regexFilterTip = $(".sidebarred > .sidebarred-main-content > .script-list#browse-script-list")
  544. ?.previousElementSibling?.appendChild?.(document.createElement("span"));
  545. if (regexFilterTip) {
  546. regexFilterTip.id = idPrefix + "regex-filter-tip";
  547. regexFilterTip.title = `[${name}] Number of scripts filtered by regex`;
  548. }
  549. function setRegexFilterTip(content) {
  550. if (regexFilterTip) {
  551. regexFilterTip.textContent = content;
  552. }
  553. }
  554. function regexFilterOne(regex, script) {
  555. const info = script.querySelector("article > h2");
  556. if (!info) return;
  557. const name = info.querySelector(".script-link").textContent;
  558. const result = regex.test(name);
  559. script.classList.toggle("regex-filtered", result);
  560. if (result) {
  561. log("Filtered:", name);
  562. }
  563. return result;
  564. }
  565. function regexFilter(regexStr) {
  566. const debug = config.get("other.debug");
  567. const scripts = $$(".script-list > li");
  568. if (regexStr === "" || scripts.length === 0) {
  569. scripts.forEach(script => script.classList.remove("regex-filtered"));
  570. setRegexFilterTip("");
  571. return;
  572. }
  573. const regex = new RegExp(regexStr, "i");
  574. let count = 0;
  575. debug && console.groupCollapsed(`[${name}] Regex filtered scripts`);
  576. scripts.forEach(script => {
  577. if (regexFilterOne(regex, script)) {
  578. count++;
  579. }
  580. });
  581. setRegexFilterTip(`Filtered: ${count}/${scripts.length}`);
  582. debug && console.groupEnd();
  583. }
  584. regexFilter(config.get("filterAndSearch.regexFilter"));
  585. // Search syntax
  586. const types = {
  587. "script": "scripts",
  588. "lib": "scripts/libraries",
  589. "library": "scripts/libraries",
  590. // "code": "scripts/code-search", // It uses a different search parameter `c` instead of `q`
  591. "user": "users"
  592. };
  593. const langs = {
  594. "js": "",
  595. "javascript": "",
  596. "css": "css",
  597. "any": "all",
  598. "all": "all"
  599. };
  600. const sorts = {
  601. "rel": "",
  602. "relevant": "",
  603. "relevance": "",
  604. "day": "daily_installs",
  605. "daily": "daily_installs",
  606. "daily_install": "daily_installs",
  607. "daily_installs": "daily_installs",
  608. "total": "total_installs",
  609. "total_install": "total_installs",
  610. "total_installs": "total_installs",
  611. "score": "ratings",
  612. "rate": "ratings",
  613. "rating": "ratings",
  614. "ratings": "ratings",
  615. "created": "created",
  616. "created_at": "created",
  617. "updated": "updated",
  618. "updated_at": "updated",
  619. "name": "name",
  620. "title": "name",
  621. };
  622. if (config.get("filterAndSearch.searchSyntax")) {
  623. function parseString(input) {
  624. // Regular expression to match key:value pairs, allowing for non-word characters in values
  625. const regex = /\b(\w+:[^\s]+)\b/g;
  626. // Extract all key:value pairs
  627. const pairs = input.match(regex) || [];
  628. // Remove the pairs from the input string
  629. const cleanedString = input.replace(regex, '').replace(/\s{2,}/g, ' ').trim();
  630.  
  631. // Convert pairs to an object
  632. const parsedPairs = pairs.reduce((acc, pair) => {
  633. const [key, value] = pair.split(':');
  634. acc[key.toLowerCase()] = value.toLowerCase(); // Case-insensitive
  635. return acc;
  636. }, {});
  637.  
  638. return { cleanedString, parsedPairs };
  639. }
  640. function processSearch(search) {
  641. const form = search.form;
  642. if (form.method !== "get") {
  643. return;
  644. }
  645. form.addEventListener("submit", (e) => {
  646. const { cleanedString, parsedPairs } = parseString(search.value);
  647. if (cleanedString === search.value) return;
  648. search.value = cleanedString;
  649. if (!parsedPairs) return;
  650. e.preventDefault();
  651. const url = new URL(form.action, window.location.href);
  652. url.searchParams.set("q", cleanedString);
  653. if (parsedPairs["site"]) { // site:site-name
  654. url.pathname = `/scripts/by-site/${parsedPairs["site"]}`;
  655. } else if (parsedPairs["type"]) { // type:type, including "script", "lib"/"library", "code", "user"
  656. const typeUrl = types[parsedPairs["type"]];
  657. if (typeUrl) {
  658. url.pathname = `/${typeUrl}`;
  659. }
  660. }
  661. if (parsedPairs["lang"]) { // lang:language
  662. const lang = langs[parsedPairs["lang"]];
  663. if (lang === "") {
  664. url.searchParams.delete("language");
  665. } else if (lang) {
  666. url.searchParams.set("language", lang);
  667. }
  668. }
  669. if (parsedPairs["sort"]) { // sort:sort-by
  670. const sort = sorts[parsedPairs["sort"]];
  671. if (sort === "" || sort === "daily_installs" && cleanedString === "") {
  672. url.searchParams.delete("sort");
  673. } else if (sort) {
  674. url.searchParams.set("sort", sort);
  675. }
  676. }
  677. window.location.href = url.href;
  678. });
  679. }
  680. const searches = $$("input[type=search][name=q]");
  681. for (const search of searches) {
  682. processSearch(search);
  683. }
  684. }
  685.  
  686. // Code blocks
  687. const codeBlocks = document.getElementsByTagName("pre");
  688. // Toolbar
  689. const toolbarEnabled = config.get("codeblocks.toolbar");
  690. if (toolbarEnabled) {
  691. async function animate(node, animation) {
  692. return new Promise((resolve, reject) => {
  693. node.classList.add("animate__animated", "animate__" + animation);
  694. if (node.getAnimations().length == 0) {
  695. node.classList.remove("animate__animated", "animate__" + animation);
  696. reject("No animation available");
  697. }
  698. node.addEventListener('animationend', e => {
  699. e.stopPropagation();
  700. node.classList.remove("animate__animated", "animate__" + animation);
  701. resolve("Animation ended");
  702. }, { once: true });
  703. });
  704. }
  705. async function transition(node, height) {
  706. return new Promise((resolve, reject) => {
  707. node.style.height = height;
  708. if (node.getAnimations().length == 0) {
  709. resolve("No transition available");
  710. }
  711. node.addEventListener('transitionend', e => {
  712. e.stopPropagation();
  713. resolve("Transition ended");
  714. }, { once: true });
  715. });
  716. }
  717. function copyCode() {
  718. const code = this.parentNode.nextElementSibling;
  719. const text = code.textContent;
  720. navigator.clipboard.writeText(text).then(() => {
  721. this.textContent = "Copied!";
  722. animate(this, "tada").then(() => {
  723. this.textContent = "Copy code";
  724. }, () => {
  725. window.setTimeout(() => {
  726. this.textContent = "Copy code";
  727. }, 1000);
  728. });
  729. });
  730. }
  731. function toggleCode() {
  732. const code = this.parentNode.nextElementSibling;
  733. if (code.style.height == "0px") {
  734. code.style.willChange = "height";
  735. transition(code, code.getAttribute("data-height")).then(() => {
  736. code.style.willChange = "";
  737. });
  738. animate(this, "fadeOut").then(() => {
  739. this.textContent = "Hide code";
  740. animate(this, "fadeIn");
  741. }, () => {
  742. this.textContent = "Hide code";
  743. });
  744. } else {
  745. code.style.willChange = "height";
  746. transition(code, "0px").then(() => {
  747. code.style.willChange = "";
  748. });
  749. animate(this, "fadeOut").then(() => {
  750. this.textContent = "Show code";
  751. animate(this, "fadeIn");
  752. }, () => {
  753. this.textContent = "Show code";
  754. });
  755. }
  756. }
  757. function createToolbar() {
  758. const toolbar = document.createElement("div");
  759. const copy = toolbar.appendChild(document.createElement("a"));
  760. const toggle = toolbar.appendChild(document.createElement("a"));
  761. copy.textContent = "Copy code";
  762. copy.className = "code-operation";
  763. copy.title = "Copy code to clipboard";
  764. copy.addEventListener("click", copyCode);
  765. toggle.textContent = "Hide code";
  766. toggle.classList.add("code-operation", "animate__fastest");
  767. toggle.title = "Toggle code display";
  768. toggle.addEventListener("click", toggleCode);
  769. // Css
  770. toolbar.className = "code-toolbar";
  771. return toolbar;
  772. }
  773. for (const codeBlock of codeBlocks) {
  774. if (codeBlock.firstChild.tagName === "CODE") {
  775. const height = getComputedStyle(codeBlock.firstChild).getPropertyValue("height");
  776. codeBlock.firstChild.style.height = height;
  777. codeBlock.firstChild.setAttribute("data-height", height);
  778. codeBlock.insertAdjacentElement("afterbegin", createToolbar());
  779. }
  780. }
  781. }
  782. // Auto hide code blocks
  783. function autoHide() {
  784. if (!toolbarEnabled) return;
  785. if (!config.get("codeblocks.autoHideCode")) {
  786. for (const code_block of codeBlocks) {
  787. const toggle = code_block.firstChild.lastChild;
  788. if (!toggle) continue;
  789. if (toggle.textContent === "Show code") {
  790. toggle.click(); // Click the toggle button
  791. }
  792. }
  793. } else {
  794. for (const codeBlock of codeBlocks) {
  795. const m = codeBlock.lastChild.textContent.match(/\n/g);
  796. const rows = m ? m.length : 0;
  797. const toggle = codeBlock.firstChild.lastChild;
  798. if (!toggle) continue;
  799. const hidden = toggle.textContent === "Show code";
  800. if (rows >= config.get("codeblocks.autoHideRows") && !hidden || rows < config.get("codeblocks.autoHideRows") && hidden) {
  801. codeBlock.firstChild.lastChild.click(); // Click the toggle button
  802. }
  803. }
  804. }
  805. }
  806. document.addEventListener("readystatechange", (e) => {
  807. if (e.target.readyState === "complete") {
  808. autoHide();
  809. }
  810. }, { once: true });
  811. // Tab size
  812. function tabSize(value) {
  813. const style = $("style#" + idPrefix + "tab-size") ?? document.head.appendChild(document.createElement("style"));
  814. style.id = idPrefix + "tab-size";
  815. style.textContent = `pre { tab-size: ${value}; }`;
  816. }
  817. tabSize(config.get("codeblocks.tabSize"));
  818. // Metadata
  819. function extractUserScriptMetadata(code) {
  820. const result = {};
  821. const userScriptRegex = /\/\/\s*=+\s*UserScript\s*=+\s*([\s\S]*?)\s*=+\s*\/UserScript\s*=+\s*/;
  822. const match = code.match(userScriptRegex);
  823. if (match) {// If the UserScript block is found
  824. const content = match[1];// Extract the content within the UserScript block
  825. const lines = content.split('\n'); // Split the content by newline
  826.  
  827. lines.forEach(line => {
  828. // Regular expression to match "// @name value" pattern
  829. const matchLine = line.trim().match(/^\/\/\s*@(\S+)\s+(.+)$/);
  830. if (matchLine) {
  831. const name = matchLine[1]; // Extract the name
  832. const value = matchLine[2]; // Extract the value
  833. switch (typeof result[name]) {
  834. case "undefined": // First occurrence
  835. result[name] = value;
  836. break;
  837. case "string": // Second occurrence
  838. result[name] = [result[name], value];
  839. break;
  840. case "object": // Third or more occurrence
  841. result[name].push(value);
  842. break;
  843. }
  844. }
  845. });
  846. }
  847. return result;
  848. }
  849. function metadata(enable) {
  850. const id = idPrefix + "metadata";
  851. const current = document.getElementById(id);
  852. if (current && !enable) {
  853. current.remove();
  854. } else if (!current && enable) {
  855. const scriptCodeBlock = document.querySelector(".code-container > pre.prettyprint.lang-js");
  856. const description = $("div#script-content");
  857. if (!window.location.pathname.endsWith("/code") || !scriptCodeBlock || !description) return;
  858. const metaBlock = document.createElement("ul");
  859. description.prepend(metaBlock);
  860. metaBlock.id = id;
  861. const script = scriptCodeBlock.querySelector("ol") ? Array.from(scriptCodeBlock.querySelectorAll("ol > li")).map(li => li.textContent).join("\n") : scriptCodeBlock.textContent;
  862. const metadata = extractUserScriptMetadata(script);
  863. const commonHosts = {
  864. GreasyFork: /^https?:\/\/update\.greasyfork\.org\/scripts\/\d+\/(?<ver>\d+)\/(?<name>.+?)\.js$/,
  865. JsDelivr: /^https?:\/\/cdn\.jsdelivr\.net\/(?<reg>\w+)\/(@[^/]+\/)?(?<name>[^@]+)@(?<ver>[^/]+)/,
  866. Cloudflare: /^https?:\/\/cdnjs\.cloudflare\.com\/ajax\/libs\/(?<name>[^/]+)\/(?<ver>[^/]+)/,
  867. };
  868. const commonRegistries = {
  869. npm: "NPM",
  870. gh: "GitHub",
  871. };
  872. // We're interested in `@grant`, `@connect`, `@require`, `@resource`
  873. const interestedMetadata = {};
  874. const interestedKeys = {
  875. grant: {
  876. brief: "Required permissions",
  877. display: (value) => {
  878. const valueCode = document.createElement("code");
  879. valueCode.textContent = value;
  880. if (value !== "none") {
  881. const valueLink = document.createElement("a");
  882. valueLink.appendChild(valueCode);
  883. valueLink.href = `https://www.tampermonkey.net/documentation.php#api:${valueCode.textContent}`;
  884. valueLink.title = `See documentation about ${valueCode.textContent}`;
  885. return valueLink;
  886. } else {
  887. return valueCode;
  888. }
  889. }
  890. },
  891. connect: {
  892. brief: "Allowed URLs to connect",
  893. display: (value) => {
  894. const valueCode = document.createElement("code");
  895. valueCode.textContent = value;
  896. return valueCode;
  897. }
  898. },
  899. require: {
  900. brief: "External libraries",
  901. display: (value) => {
  902. const valueLink = document.createElement("a");
  903. valueLink.href = value;
  904. valueLink.textContent = value;
  905. for (const [host, regex] of Object.entries(commonHosts)) {
  906. const match = value.match(regex);
  907. if (match) {
  908. const { name, ver, reg } = match.groups;
  909. const optionalRegistry = commonRegistries[reg] ? `${commonRegistries[reg]} on ` : "";
  910. valueLink.textContent = `${decodeURIComponent(name)}@${ver} (${optionalRegistry}${host})`;
  911. break;
  912. }
  913. }
  914. return valueLink;
  915. }
  916. },
  917. resource: {
  918. brief: "External resources",
  919. display: (value) => {
  920. const valueCode = document.createElement("code");
  921. const [name, link] = value.split(" ");
  922. const valueLink = document.createElement("a");
  923. valueLink.appendChild(valueCode);
  924. valueLink.href = link.trim();
  925. valueCode.textContent = name.trim();
  926. return valueLink;
  927. }
  928. }
  929. };
  930. for (const key in interestedKeys) {
  931. const values = metadata[key] ?? [];
  932. interestedMetadata[key] = Array.isArray(values) ? values : [values];
  933. }
  934. log("Interested Metadata:", interestedMetadata);
  935. // Display
  936. for (const [key, values] of Object.entries(interestedMetadata)) {
  937. const keyInfo = interestedKeys[key];
  938. const li = metaBlock.appendChild(document.createElement("li"));
  939. const keyLink = li.appendChild(document.createElement("a"));
  940. keyLink.href = `https://www.tampermonkey.net/documentation.php#meta:${key}`;
  941. keyLink.title = keyInfo.brief;
  942. keyLink.textContent = `@${key}`;
  943. const separator = li.appendChild(document.createElement("span"));
  944. separator.textContent = ": ";
  945. for (const value of values) {
  946. li.appendChild(keyInfo.display(value));
  947. const separator = li.appendChild(document.createElement("span"));
  948. separator.textContent = ", ";
  949. }
  950. if (values.length > 0) {
  951. li.lastChild.remove(); // Remove the last separator
  952. } else {
  953. li.appendChild(document.createTextNode("none"));
  954. }
  955. }
  956. }
  957. }
  958. metadata(config.get("codeblocks.metadata"));
  959.  
  960. // Display
  961. // Flat layout
  962. function flatLayout(enable) {
  963. const meta_orig = $("#script-info > #script-content .script-meta-block");
  964. const meta_mod = $("#script-info > .script-meta-block");
  965. if (enable && meta_orig) {
  966. const header = $("#script-info > header");
  967. header.before(meta_orig);
  968. } else if (!enable && meta_mod) {
  969. const additional = $("#script-info > #script-content #additional-info");
  970. additional.before(meta_mod);
  971. }
  972. }
  973. flatLayout(config.get("display.flatLayout"));
  974. // Always show notification
  975. function alwaysShowNotification(enable) {
  976. const nav = $("#nav-user-info");
  977. const profile = nav?.querySelector(".user-profile-link");
  978. const existing = nav.querySelector(".notification-widget");
  979. if (!nav || !profile || existing && existing.textContent !== "0") return; // There's unread notification or user is not logged in
  980. if (enable && !existing) {
  981. const notification = nav.insertBefore(document.createElement("a"), profile);
  982. notification.className = "notification-widget";
  983. notification.textContent = "0";
  984. notification.href = profile.querySelector("a").href + "/notifications";
  985. } else if (!enable && existing) {
  986. existing.remove();
  987. }
  988. }
  989. alwaysShowNotification(config.get("display.alwaysShowNotification"));
  990.  
  991. // Credenials
  992. // Auto login
  993. async function login(email, password) {
  994. log("Login:", email, "*".repeat(password.length));
  995. const initReq = await fetch("/users/sign_in", {
  996. method: "GET",
  997. credentials: "same-origin",
  998. headers: {
  999. "Accept": "text/html",
  1000. }
  1001. });
  1002. const text = await initReq.text();
  1003. const parser = new DOMParser();
  1004. const doc = parser.parseFromString(text, "text/html");
  1005. const fd = new FormData(doc.querySelector("form#new_user"));
  1006. fd.set("user[email]", email);
  1007. fd.set("user[password]", password);
  1008. fd.set("user[remember_me]", "1");
  1009.  
  1010. const loginReq = await fetch(initReq.url, {
  1011. method: "POST",
  1012. credentials: "same-origin",
  1013. body: fd,
  1014. headers: {
  1015. "Accept": "text/html",
  1016. }
  1017. });
  1018. log("Login request:", loginReq);
  1019. return loginReq.ok;
  1020. }
  1021. function autoLogin(mode) {
  1022. if (mode === 0 || $("#nav-user-info .user-profile-link")) return; // Not enabled or already logged in
  1023. if (mode === 1 && !$("#home-script-nav")) return; // Not on the home page
  1024. // Validate credentials
  1025. const email = config.get("credentials.email");
  1026. const password = config.get("credentials.password");
  1027. if (!email || !password || !email.includes("@")) {
  1028. log("Invalid credentials - skipping auto login");
  1029. return;
  1030. }
  1031. // Login
  1032. const hint = $("#nav-user-info > .sign-in-link > a");
  1033. hint.textContent = "[GFE] Logging in...";
  1034. hint.title = `[${name}] Auto login in progress`;
  1035. hint.setAttribute("href", "javascript:void(0)");
  1036. if (login(email, password)) {
  1037. log("Auto login successful, will refresh in a moment");
  1038. hint.textContent = "[GFE] Logged in, refreshing...";
  1039. hint.title = `[${name}] Auto login successful, will refresh in a moment`;
  1040. setTimeout(() => {
  1041. location.reload();
  1042. }, 3000);
  1043. } else {
  1044. log("Login failed, auto login disabled");
  1045. hint.textContent = "[GFE] Login failed";
  1046. hint.title = `[${name}] Login failed, auto login disabled`;
  1047. config.set("credentials.autoLogin", 0);
  1048. }
  1049. }
  1050. autoLogin(config.get("credentials.autoLogin"));
  1051. // Capture credentials
  1052. function onSubmit(e) {
  1053. log("Login attempt detected");
  1054. e.preventDefault(); // DEBUG
  1055. const fd = new FormData(e.target);
  1056. // Extract email and password
  1057. const email = fd.get("user[email]");
  1058. const password = fd.get("user[password]");
  1059. // If both are present...
  1060. if (email && password) {
  1061. // ...then capture the credentials
  1062. log("Captured credentials");
  1063. config.set("credentials.email", email);
  1064. config.set("credentials.password", password);
  1065. }
  1066. }
  1067. let captureEnabled = false;
  1068. function captureCredentials(enable) {
  1069. if (!location.pathname.endsWith("/users/sign_in") || captureEnabled === enable) return;
  1070. const form = $("form#new_user");
  1071. if (!form) return;
  1072. if (enable) {
  1073. form.addEventListener("submit", onSubmit);
  1074. } else {
  1075. form.removeEventListener("submit", onSubmit);
  1076. }
  1077. captureEnabled = enable;
  1078. }
  1079. captureCredentials(config.get("credentials.captureCredentials"));
  1080. // Other
  1081. // Short link
  1082. function shortLink(enable) {
  1083. const description = $("div#script-content");
  1084. const url = window.location.href;
  1085. const scriptId = url.match(/\/scripts\/(\d+)/)?.[1];
  1086. if (!scriptId || !description) return;
  1087. const id = idPrefix + "short-link";
  1088. const current = document.getElementById(id);
  1089. if (current && !enable) {
  1090. current.remove();
  1091. } else if (!current && enable) {
  1092. const short = `https://gf.qytechs.cn/scripts/${scriptId}`;
  1093. const p = description.insertAdjacentElement("beforebegin", document.createElement("p"));
  1094. p.id = id;
  1095. p.textContent = "Short link: ";
  1096. const link = p.appendChild(document.createElement("a"));
  1097. link.href = short;
  1098. link.textContent = short;
  1099. const copy = p.appendChild(document.createElement("a"));
  1100. copy.textContent = "(Copy)";
  1101. copy.style.marginLeft = "1em";
  1102. copy.style.cursor = "pointer";
  1103. copy.title = "Copy short link to clipboard";
  1104. copy.addEventListener("click", () => {
  1105. if (copy.textContent === "(Copied!)") return;
  1106. navigator.clipboard.writeText(short).then(() => {
  1107. copy.textContent = "(Copied!)";
  1108. window.setTimeout(() => {
  1109. copy.textContent = "(Copy)";
  1110. }, 1000);
  1111. });
  1112. });
  1113. }
  1114. }
  1115. shortLink(config.get("other.shortLink"));
  1116. // Alternative URLs for library
  1117. function alternativeURLs(enable) {
  1118. if ($(".remove-attachments") || !$("div#script-content") || $("div#script-content > div#install-area")) return; // Not a library
  1119. const id = idPrefix + "lib-alternative-url";
  1120. const current = document.getElementById(id);
  1121. if (current && !enable) {
  1122. current.remove();
  1123. } else if (!current && enable) {
  1124. const description = $("div#script-content > p");
  1125. const trim = "// @require ";
  1126. const text = description?.querySelector("code")?.textContent;
  1127. if (!text || !text.startsWith(trim)) return; // Found no URL
  1128. const url = text.slice(trim.length);
  1129. const parts = url.split("/");
  1130. const scriptId = parts[4];
  1131. const scriptVersion = parts[5];
  1132. const fileName = parts[6];
  1133. const URLs = [
  1134. [`// @require https://update.gf.qytechs.cn/scripts/${scriptId}/${fileName}`, "Latest version"],
  1135. [`// @require https://gf.qytechs.cn/scripts/${scriptId}/code/${fileName}?version=${scriptVersion}`, "Current version (Legacy)"],
  1136. [`// @require https://gf.qytechs.cn/scripts/${scriptId}/code/${fileName}`, "Latest version (Legacy)"],
  1137. ];
  1138.  
  1139. const detail = document.createElement("p").appendChild(document.createElement("details"));
  1140. description.after(detail.parentElement);
  1141. detail.parentElement.id = id;
  1142. detail.appendChild(document.createElement("summary")).textContent = "Alternative URLs";
  1143. const list = detail.appendChild(document.createElement("ul"));
  1144. for (const [url, text] of URLs) {
  1145. const link = list.appendChild(document.createElement("li")).appendChild(document.createElement("code"));
  1146. link.textContent = url;
  1147. link.title = text;
  1148. }
  1149. }
  1150. }
  1151. alternativeURLs(config.get("other.libAlternativeUrl"));
  1152. // Image proxy
  1153. if (config.get("other.imageProxy")) {
  1154. const PROXY = "https://wsrv.nl/?url=";
  1155. const images = $$("a[href^='/rails/active_storage/blobs/redirect/'] > img[src^='https://greasyfork.']");
  1156. for (const img of images) {
  1157. img.src = PROXY + img.src;
  1158. const link = img.parentElement;
  1159. link.href = PROXY + link.href;
  1160. }
  1161. }
  1162. // Lazy image
  1163. if (config.get("other.lazyImage")) {
  1164. const images = $$(".user-content img");
  1165. for (const image of images) {
  1166. image.loading = "lazy";
  1167. }
  1168. }
  1169.  
  1170. // Initialize css
  1171. for (const prop in dynamicStyles) {
  1172. cssHelper(prop, config.get(prop));
  1173. }
  1174. for (const prop in enumStyles) {
  1175. enumStyleHelper(prop, config.get(prop));
  1176. }
  1177. // Dynamically respond to config changes
  1178. const callbacks = {
  1179. "filterAndSearch.shortcut": shortcut,
  1180. "filterAndSearch.regexFilter": regexFilter,
  1181. "codeblocks.autoHideCode": autoHide,
  1182. "codeblocks.autoHideRows": autoHide,
  1183. "codeblocks.tabSize": tabSize,
  1184. "codeblocks.metadata": metadata,
  1185. "display.flatLayout": flatLayout,
  1186. "display.alwaysShowNotification": alwaysShowNotification,
  1187. "credentials.captureCredentials": captureCredentials,
  1188. "other.shortLink": shortLink,
  1189. "other.libAlternativeUrl": alternativeURLs,
  1190. };
  1191. config.addEventListener("set", e => {
  1192. if (e.detail.prop in dynamicStyles) {
  1193. cssHelper(e.detail.prop, e.detail.after);
  1194. }
  1195. if (e.detail.prop in enumStyles) {
  1196. enumStyleHelper(e.detail.prop, e.detail.after);
  1197. }
  1198. const callback = callbacks[e.detail.prop];
  1199. if (callback && (e.detail.before !== e.detail.after)) {
  1200. callback(e.detail.after);
  1201. }
  1202. });
  1203. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址