Greasy Fork镜像++

添加各种功能并改善 Greasy Fork镜像 体验

  1. // ==UserScript==
  2. // @name Greasy Fork镜像++
  3. // @namespace https://github.com/iFelix18
  4. // @version 3.3.4
  5. // @author CY Fung <https://gf.qytechs.cn/users/371179> & Davide <iFelix18@protonmail.com>
  6. // @icon https://www.google.com/s2/favicons?domain=https://gf.qytechs.cn
  7. // @description Adds various features and improves the Greasy Fork镜像 experience
  8. // @description:de Fügt verschiedene Funktionen hinzu und verbessert das Greasy Fork镜像-Erlebnis
  9. // @description:es Agrega varias funciones y mejora la experiencia de Greasy Fork镜像
  10. // @description:fr Ajoute diverses fonctionnalités et améliore l'expérience Greasy Fork镜像
  11. // @description:it Aggiunge varie funzionalità e migliora l'esperienza di Greasy Fork镜像
  12. // @description:ru Добавляет различные функции и улучшает работу с Greasy Fork镜像
  13. // @description:zh-CN 添加各种功能并改善 Greasy Fork镜像 体验
  14. // @description:zh-TW 加入多種功能並改善Greasy Fork镜像的體驗
  15. // @description:ja Greasy Fork镜像の体験を向上させる様々な機能を追加
  16. // @description:ko Greasy Fork镜像 경험을 향상시키고 다양한 기능을 추가
  17. // @copyright 2023, CY Fung (https://gf.qytechs.cn/users/371179); 2021, Davide (https://github.com/iFelix18)
  18. // @license MIT
  19. // @require https://fastly.jsdelivr.net/gh/sizzlemctwizzle/GM_config@06f2015c04db3aaab9717298394ca4f025802873/gm_config.min.js
  20. // @require https://fastly.jsdelivr.net/npm/@violentmonkey/shortcut@1.4.1/dist/index.min.js
  21. // @require https://fastly.jsdelivr.net/gh/cyfung1031/userscript-supports@3fa07109efca28a21094488431363862ccd52d7c/library/WinComm.min.js
  22. // @match *://gf.qytechs.cn/*
  23. // @match *://sleazyfork.org/*
  24. // @match *://gf.qytechs.cn/*
  25. // @match *://api.gf.qytechs.cn/*
  26. // @match *://api.sleazyfork.org/*
  27. // @match *://api.gf.qytechs.cn/*
  28. // @connect gf.qytechs.cn
  29. // @connect sleazyfork.org
  30. // @connect gf.qytechs.cn
  31. // @compatible chrome
  32. // @compatible edge
  33. // @compatible firefox
  34. // @compatible safari
  35. // @compatible brave
  36. // @grant GM.deleteValue
  37. // @grant GM.getValue
  38. // @grant GM.notification
  39. // @grant GM.registerMenuCommand
  40. // @grant GM.setValue
  41. // @grant unsafeWindow
  42. // @run-at document-start
  43. // @inject-into content
  44. // ==/UserScript==
  45.  
  46. /* global GM_config, VM, GM, WinComm */
  47.  
  48. const isInIframe = window !== top;
  49.  
  50. /**
  51. * @typedef { typeof import("./library/WinComm.js") } WinComm
  52. */
  53.  
  54. // console.log(GM)
  55.  
  56. /** @type {WinComm} */
  57. const WinComm = this.WinComm;
  58.  
  59. // -------- UU Fucntion - original code: https://fastly.jsdelivr.net/npm/@ifelix18/utils@6.5.0/lib/index.min.js --------
  60. // optimized by CY Fung to remove $ dependency and observe creation
  61. const UU = isInIframe || (function () {
  62. const scriptName = GM.info.script.name; // not name_i18n
  63. const scriptVersion = GM.info.script.version;
  64. const authorMatch = /^(.*?)\s<\S[^\s@]*@\S[^\s.]*\.\S+>$/.exec(GM.info.script.author);
  65. const author = authorMatch ? authorMatch[1] : GM.info.script.author;
  66. let scriptId = scriptName.toLowerCase().replace(/\s/g, "-");
  67. let loggingEnabled = false;
  68.  
  69. const log = (message) => {
  70. if (loggingEnabled) {
  71. console.log(`${scriptName}:`, message);
  72. }
  73. };
  74.  
  75. const error = (message) => {
  76. console.error(`${scriptName}:`, message);
  77. };
  78.  
  79. const warn = (message) => {
  80. console.warn(`${scriptName}:`, message);
  81. };
  82.  
  83. const alert = (message) => {
  84. window.alert(`${scriptName}: ${message}`);
  85. };
  86.  
  87. /** @param {string} text */
  88. const short = (text, length) => {
  89. const s = text.split(" ");
  90. const l = Number(length);
  91. return s.length > l
  92. ? `${s.slice(0, l).join(" ")} [...]`
  93. : text;
  94. };
  95.  
  96. const addStyle = (css) => {
  97. const head = document.head || document.querySelector("head");
  98. const style = document.createElement("style");
  99. style.textContent = css;
  100. head.appendChild(style);
  101. };
  102.  
  103. const init = async (options = {}) => {
  104. scriptId = options.id || scriptId;
  105. loggingEnabled = typeof options.logging === "boolean" ? options.logging : false;
  106. console.info(
  107. `%c${scriptName}\n%cv${scriptVersion}${author ? ` by ${author}` : ""} is running!`,
  108. "color:red;font-weight:700;font-size:18px;text-transform:uppercase",
  109. ""
  110. );
  111. };
  112.  
  113. return {
  114. init,
  115. log,
  116. error,
  117. warn,
  118. alert,
  119. short,
  120. addStyle
  121. };
  122. })();
  123.  
  124. // -------- UU Fucntion - original code: https://fastly.jsdelivr.net/npm/@ifelix18/utils@6.5.0/lib/index.min.js --------
  125.  
  126.  
  127. const mWindow = isInIframe || (() => {
  128.  
  129.  
  130. const fields = {
  131. hideBlacklistedScripts: {
  132. label: 'Hide blacklisted scripts:<br><span>Choose which lists to activate in the section below, press <b>Ctrl + Alt + B</b> to show Blacklisted scripts</span>',
  133. section: ['Features'],
  134. labelPos: 'right',
  135. type: 'checkbox',
  136. default: true
  137. },
  138. hideHiddenScript: {
  139. label: 'Hide scripts:<br><span>Add a button to hide the script<br>See and edit the list of hidden scripts below, press <b>Ctrl + Alt + H</b> to show Hidden script',
  140. labelPos: 'right',
  141. type: 'checkbox',
  142. default: true
  143. },
  144. showInstallButton: {
  145. label: 'Install button:<br><span>Add to the scripts list a button to install the script directly</span>',
  146. labelPos: 'right',
  147. type: 'checkbox',
  148. default: true
  149. },
  150. showTotalInstalls: {
  151. label: 'Installations:<br><span>Shows the number of daily and total installations on the user profile</span>',
  152. labelPos: 'right',
  153. type: 'checkbox',
  154. default: true
  155. },
  156. milestoneNotification: {
  157. label: 'Milestone notifications:<br><span>Get notified whenever your total installs got over any of these milestone<br>Separate milestones with a comma, leave blank to turn off notifications</span>',
  158. labelPos: 'left',
  159. type: 'text',
  160. title: 'Separate milestones with a comma!',
  161. size: 150,
  162. default: '10, 100, 500, 1000, 2500, 5000, 10000, 100000, 1000000'
  163. },
  164. nonLatins: {
  165. label: 'Non-Latin:<br><span>This list blocks all scripts with non-Latin characters in the title/description</span>',
  166. section: ['Lists'],
  167. labelPos: 'right',
  168. type: 'checkbox',
  169. default: false // not true
  170. },
  171. blacklist: {
  172. label: 'Blacklist:<br><span>A "non-opinionable" list that blocks all scripts with specific words in the title/description, references to "bots", "cheats" and some online game sites, and other "bullshit"</span>',
  173. labelPos: 'right',
  174. type: 'checkbox',
  175. default: true
  176. },
  177. customBlacklist: {
  178. label: 'Custom Blacklist:<br><span>Personal blacklist defined by a set of unwanted words<br>Separate unwanted words with a comma (example: YouTube, Facebook, pizza), leave blank to disable this list</span>',
  179. labelPos: 'left',
  180. type: 'text',
  181. title: 'Separate unwanted words with a comma!',
  182. size: 150,
  183. default: ''
  184. },
  185. hiddenList: {
  186. label: 'Hidden Scripts:<br><span>Block individual undesired scripts by their unique IDs<br>Separate IDs with a comma</span>',
  187. labelPos: 'left',
  188. type: 'textarea',
  189. title: 'Separate IDs with a comma!',
  190. default: '',
  191. save: false
  192. },
  193. hideRecentUsersWithin: {
  194. label: 'Hide Recent Users:<br><span>Hide new regeistered users within the last N hours - to avoid seeing comments from spam accounts</span>',
  195. labelPos: 'left',
  196. type: 'text',
  197. title: 'Number only. 0 means disabled. maximum is 168. (Suggested value: 48)',
  198. default: '0',
  199. size: 150
  200. },
  201. logging: {
  202. label: 'Logging',
  203. section: ['Developer options'],
  204. labelPos: 'right',
  205. type: 'checkbox',
  206. default: false
  207. },
  208. debugging: {
  209. label: 'Debugging',
  210. labelPos: 'right',
  211. type: 'checkbox',
  212. default: false
  213. }
  214. }
  215.  
  216. const logo = ''
  217.  
  218. const locales = { /* cSpell: disable */
  219. de: {
  220. downgrade: 'Auf zurückstufen',
  221. hide: '❌ Dieses skript ausblenden',
  222. install: 'Installieren',
  223. notHide: '✔️ Dieses skript nicht ausblenden',
  224. milestone: 'Herzlichen Glückwunsch, Ihre Skripte haben den Meilenstein von insgesamt $1 Installationen überschritten!',
  225. reinstall: 'Erneut installieren',
  226. update: 'Auf aktualisieren'
  227. },
  228. en: {
  229. downgrade: 'Downgrade to',
  230. hide: '❌ Hide this script',
  231. install: 'Install',
  232. notHide: '✔️ Not hide this script',
  233. milestone: 'Congrats, your scripts got over the milestone of $1 total installs!',
  234. reinstall: 'Reinstall',
  235. update: 'Update to'
  236. },
  237. es: {
  238. downgrade: 'Degradar a',
  239. hide: '❌ Ocultar este script',
  240. install: 'Instalar',
  241. notHide: '✔️ No ocultar este script',
  242. milestone: '¡Felicidades, sus scripts superaron el hito de $1 instalaciones totales!',
  243. reinstall: 'Reinstalar',
  244. update: 'Actualizar a'
  245. },
  246. fr: {
  247. downgrade: 'Revenir à',
  248. hide: '❌ Cacher ce script',
  249. install: 'Installer',
  250. notHide: '✔️ Ne pas cacher ce script',
  251. milestone: 'Félicitations, vos scripts ont franchi le cap des $1 installations au total!',
  252. reinstall: 'Réinstaller',
  253. update: 'Mettre à'
  254. },
  255. it: {
  256. downgrade: 'Riporta a',
  257. hide: '❌ Nascondi questo script',
  258. install: 'Installa',
  259. notHide: '✔️ Non nascondere questo script',
  260. milestone: 'Congratulazioni, i tuoi script hanno superato il traguardo di $1 installazioni totali!',
  261. reinstall: 'Reinstalla',
  262. update: 'Aggiorna a'
  263. },
  264. ru: {
  265. downgrade: 'Откатить до',
  266. hide: '❌ Скрыть этот скрипт',
  267. install: 'Установить',
  268. notHide: '✔️ Не скрывать этот сценарий',
  269. milestone: 'Поздравляем, ваши скрипты преодолели рубеж в $1 установок!',
  270. reinstall: 'Переустановить',
  271. update: 'Обновить до'
  272. },
  273. 'zh-CN': {
  274. downgrade: '降级到',
  275. hide: '❌ 隐藏此脚本',
  276. install: '安装',
  277. notHide: '✔️ 不隐藏此脚本',
  278. milestone: '恭喜,您的脚本超过了 $1 次总安装的里程碑!',
  279. reinstall: '重新安装',
  280. update: '更新到'
  281. },
  282. 'zh-TW': {
  283. downgrade: '降級至',
  284. hide: '❌ 隱藏此腳本',
  285. install: '安裝',
  286. notHide: '✔️ 不隱藏此腳本',
  287. milestone: '恭喜,您的腳本安裝總數已超過 $1!',
  288. reinstall: '重新安裝',
  289. update: '更新至'
  290. },
  291. 'ja': {
  292. downgrade: 'ダウングレードする',
  293. hide: '❌ このスクリプトを隠す',
  294. install: 'インストール',
  295. notHide: '✔️ このスクリプトを隠さない',
  296. milestone: 'おめでとうございます、あなたのスクリプトの合計インストール回数が $1 を超えました!',
  297. reinstall: '再インストール',
  298. update: '更新する'
  299. },
  300. 'ko': {
  301. downgrade: '다운그레이드하기',
  302. hide: '❌ 이 스크립트 숨기기',
  303. install: '설치',
  304. notHide: '✔️ 이 스크립트 숨기지 않기',
  305. milestone: '축하합니다, 스크립트의 총 설치 횟수가 $1을 넘었습니다!',
  306. reinstall: '재설치',
  307. update: '업데이트하기'
  308. }
  309.  
  310. };
  311.  
  312. const blacklist = [
  313. '\\bagar((\\.)?io)?\\b', '\\bagma((\\.)?io)?\\b', '\\baimbot\\b', '\\barras((\\.)?io)?\\b', '\\bbot(s)?\\b',
  314. '\\bbubble((\\.)?am)?\\b', '\\bcheat(s)?\\b', '\\bdiep((\\.)?io)?\\b', '\\bfreebitco((\\.)?in)?\\b', '\\bgota((\\.)?io)?\\b',
  315. '\\bhack(s)?\\b', '\\bkrunker((\\.)?io)?\\b', '\\blostworld((\\.)?io)?\\b', '\\bmoomoo((\\.)?io)?\\b', '\\broblox(\\.com)?\\b',
  316. '\\bshell\\sshockers\\b', '\\bshellshock((\\.)?io)?\\b', '\\bshellshockers\\b', '\\bskribbl((\\.)?io)?\\b', '\\bslither((\\.)?io)?\\b',
  317. '\\bsurviv((\\.)?io)?\\b', '\\btaming((\\.)?io)?\\b', '\\bvenge((\\.)?io)?\\b', '\\bvertix((\\.)?io)?\\b', '\\bzombs((\\.)?io)?\\b',
  318. // '\\p{Extended_Pictographic}'
  319. ];
  320.  
  321.  
  322. const settingsCSS = `
  323.  
  324. /*
  325. #greasyfork-plus label::before {
  326. content:'';
  327. display:block;
  328. position:absolute;
  329. left:0;
  330. right:0;
  331. top:0;
  332. bottom:0;
  333. z-index:1;
  334. }
  335. #greasyfork-plus label {
  336. position:relative;
  337. z-index:0;
  338. }
  339. */
  340.  
  341. html {
  342. color: #222;
  343. background: #f9f9f9;
  344. }
  345.  
  346. #greasyfork-plus{
  347. --config-var-display: flex;
  348. }
  349. #greasyfork-plus * {
  350. font-family:Open Sans,sans-serif,Segoe UI Emoji !important;
  351. font-size:12px
  352. }
  353. #greasyfork-plus .section_header[class] {
  354. background-color:#670000;
  355. background-image:linear-gradient(#670000,#900);
  356. border:1px solid transparent;
  357. color:#fff
  358. }
  359. #greasyfork-plus .field_label[class]{
  360. margin-bottom:4px
  361. }
  362. #greasyfork-plus .field_label[class] span{
  363. font-size:95%;
  364. font-style:italic;
  365. opacity:.8;
  366. }
  367. #greasyfork-plus .field_label[class] b{
  368. color:#670000
  369. }
  370. #greasyfork-plus_logging_var[class],
  371. #greasyfork-plus_debugging_var[class] {
  372. --config-var-display: inline-flex;
  373. }
  374. #greasyfork-plus #greasyfork-plus_logging_var label.field_label[class],
  375. #greasyfork-plus #greasyfork-plus_debugging_var label.field_label[class] {
  376. margin-bottom:0;
  377. align-self: center;
  378. }
  379. #greasyfork-plus .config_var[class]{
  380. display:var(--config-var-display);
  381. position: relative;
  382. }
  383. #greasyfork-plus_customBlacklist_var[class],
  384. #greasyfork-plus_hiddenList_var[class],
  385. #greasyfork-plus_milestoneNotification_var[class],
  386. #greasyfork-plus_hideRecentUsersWithin_var[class]{
  387. flex-direction:column;
  388. margin-left:21px;
  389. }
  390.  
  391. #greasyfork-plus_customBlacklist_var[class]::before,
  392. #greasyfork-plus_hiddenList_var[class]::before,
  393. #greasyfork-plus_milestoneNotification_var[class]::before,
  394. #greasyfork-plus_hideRecentUsersWithin_var[class]::before{
  395. /* content: "◉"; */
  396. content: "◎";
  397. position: absolute;
  398. left: auto;
  399. top: auto;
  400. margin-left: -16px;
  401. }
  402. #greasyfork-plus_field_customBlacklist[class],
  403. #greasyfork-plus_field_milestoneNotification[class]{
  404. flex:1;
  405. }
  406. #greasyfork-plus_field_hiddenList[class]{
  407. box-sizing:border-box;
  408. overflow:hidden;
  409. resize:none;
  410. width:100%
  411. }
  412.  
  413. body > #greasyfork-plus_wrapper:only-child {
  414. box-sizing: border-box;
  415. overflow: auto;
  416. max-height: calc(100vh - 72px);
  417. padding: 12px;
  418. /* overflow: auto; */
  419. scrollbar-gutter: both-edges;
  420. background: rgba(127,127,127,0.05);
  421. border: 1px solid rgba(127,127,127,0.5);
  422. }
  423.  
  424. #greasyfork-plus_wrapper > #greasyfork-plus_buttons_holder:last-child {
  425. position: fixed;
  426. bottom: 0;
  427. right: 0;
  428. margin: 0 12px 6px 0;
  429. }
  430.  
  431. #greasyfork-plus .saveclose_buttons[class] {
  432. padding: 4px 14px;
  433. margin: 6px;
  434. }
  435. #greasyfork-plus .section_header_holder#greasyfork-plus_section_2[class] {
  436. position: fixed;
  437. left: 0;
  438. bottom: 0;
  439. margin: 8px;
  440. }
  441. #greasyfork-plus .section_header#greasyfork-plus_section_header_2[class] {
  442. background: #000;
  443. color: #eee;
  444. }
  445.  
  446. #greasyfork-plus_header[class]{
  447. font-size: 16pt;
  448. font-weight: bold;
  449. }
  450.  
  451. `;
  452.  
  453. const pageCSS = `
  454.  
  455. .script-list li.blacklisted{
  456. display:none;
  457. background:#321919;
  458. color:#e8e6e3
  459. }
  460. .script-list li.hidden{
  461. display:none;
  462. background:#321932;
  463. color:#e8e6e3
  464. }
  465. .script-list li.blacklisted a:not(.install-link),.script-list li.hidden a:not(.install-link){
  466. color:#ff8484
  467. }
  468. #script-info.hidden,#script-info.hidden .user-content{
  469. background:#321932;
  470. color:#e8e6e3
  471. }
  472. #script-info.hidden a:not(.install-link):not(.install-help-link){
  473. color:#ff8484
  474. }
  475. #script-info.hidden code{
  476. background-color:transparent
  477. }
  478. html {
  479. --block-btn-color:#111;
  480. --block-btn-bgcolor:#eee;
  481. }
  482. #script-info.hidden, #script-info.hidden .user-content {
  483. --block-btn-color:#eee;
  484. --block-btn-bgcolor:#111;
  485. }
  486.  
  487. [style-54998]{
  488. float:right;
  489. font-size: 70%;
  490. text-decoration:none;
  491. }
  492.  
  493. [style-16377]{
  494. cursor:pointer;
  495. font-size:70%;
  496. white-space:nowrap;
  497. border: 1px solid #888;
  498. background: var(--block-btn-bgcolor, #eee);
  499. color: var(--block-btn-color);
  500. border-radius: 4px;
  501. padding: 0px 6px;
  502. margin: 0 8px;
  503. }
  504. [style-77329] {
  505. cursor: pointer;
  506. margin-left: 1ex;
  507. white-space: nowrap;
  508. float: right;
  509. border: 1px solid #888;
  510. background: var(--block-btn-bgcolor, #eee);
  511. color: var(--block-btn-color);
  512. border-radius: 4px;
  513. padding: 0px 6px;
  514. }
  515.  
  516. a#hyperlink-35389,
  517. a#hyperlink-40361,
  518. a#hyperlink-35389:visited,
  519. a#hyperlink-40361:visited,
  520. a#hyperlink-35389:hover,
  521. a#hyperlink-40361:hover,
  522. a#hyperlink-35389:focus,
  523. a#hyperlink-40361:focus,
  524. a#hyperlink-35389:active,
  525. a#hyperlink-40361:active {
  526.  
  527. border: none !important;
  528. outline: none !important;
  529. box-shadow: none !important;
  530. appearance: none !important;
  531. background: none !important;
  532. color:inherit !important;
  533. }
  534.  
  535. a#hyperlink-35389{
  536. opacity: var(--hyperlink-blacklisted-option-opacity);
  537.  
  538. }
  539. a#hyperlink-40361{
  540. opacity: var(--hyperlink-hidden-option-opacity);
  541. }
  542.  
  543.  
  544. html {
  545.  
  546. --hyperlink-blacklisted-option-opacity: 0.5;
  547. --hyperlink-hidden-option-opacity: 0.5;
  548. }
  549.  
  550.  
  551. .list-option.list-current[class] > a[href] {
  552.  
  553. text-decoration:none;
  554. }
  555.  
  556. html {
  557. --blacklisted-display: none;
  558. --hidden-display: none;
  559. }
  560.  
  561. [blacklisted-shown] {
  562. --blacklisted-display: list-item;
  563. --hyperlink-blacklisted-option-opacity: 1;
  564. }
  565. [hidden-shown] {
  566. --hidden-display: list-item;
  567. --hyperlink-hidden-option-opacity: 1;
  568. }
  569.  
  570. .script-list li.blacklisted{
  571. display: var(--blacklisted-display);
  572.  
  573. }
  574.  
  575. .script-list li.hidden{
  576. display: var(--hidden-display);
  577.  
  578. }
  579.  
  580. .install-link.install-status-checking,
  581. .install-link.install-status-checking:visited,
  582. .install-link.install-status-checking:active,
  583. .install-link.install-status-checking:hover,
  584. .install-help-link.install-status-checking {
  585. background-color: #405458;
  586. }
  587.  
  588. div.previewable{
  589. display: flex;
  590. flex-direction: column;
  591. }
  592. .script-version-ainfo-span {
  593. align-self:end;
  594. font-size: 90%;
  595. padding: 4px 8px;
  596. margin: 0;
  597. }
  598. [style*="display:"] + .script-version-ainfo-span{
  599. display: none;
  600. }
  601.  
  602.  
  603. /* Greasy Fork镜像 Enhance - Flat Layout */
  604.  
  605. [greasyfork-enhance-k37*="|flat-layout|"] ol.script-list > li > article > h2 {
  606. width: 0;
  607. flex-grow: 1;
  608. flex-basis: 60%;
  609. }
  610.  
  611. [greasyfork-enhance-k37*="|flat-layout|"] ol.script-list > li > article > div.script-meta-block {
  612. width: auto;
  613. flex-basis: 40%;
  614. flex-shrink: 0;
  615. flex-grow: 0;
  616. }
  617.  
  618. [greasyfork-enhance-k37*="|flat-layout|"] .script-list li:not(.ad-entry) {
  619. padding: 1em;
  620. margin: 0;
  621. }
  622.  
  623. [greasyfork-enhance-k37*="|flat-layout|"] .script-list li:not(.ad-entry) article {
  624. padding: 0;
  625. margin: 0;
  626. }
  627.  
  628. [greasyfork-enhance-k37*="|flat-layout|"] #script-info div.script-meta-block + #additional-info {
  629.  
  630. max-width: calc( 100% - 340px );
  631. min-height: 300px;
  632. box-sizing: border-box;
  633. }
  634.  
  635. [greasyfork-enhance-k37*="|basic|"] ul.outline {
  636. margin-bottom: -99vh;
  637.  
  638. }
  639.  
  640. .discussion-list .hidden {
  641. display: none;
  642. }
  643.  
  644. /* Greasy Fork镜像 Empty Ad Block */
  645. .ethical-ads-text[class]:empty {
  646. min-height: unset;
  647. }
  648.  
  649.  
  650. /* additional css */
  651.  
  652. .discussion-item-by-recent-user{
  653. opacity: 0.2;
  654. }
  655.  
  656. .discussion-list-item {
  657. position: relative;
  658. }
  659.  
  660. .discussion-list-item .discussion-meta .discussion-meta-item{
  661. display: flex;
  662. flex-direction: row;
  663. flex-wrap: nowrap;
  664. align-items: center;
  665. gap: 4px;
  666.  
  667. }
  668.  
  669. .discussion-list-item .discussion-meta .discussion-meta-item:last-of-type .discussion-meta-item{
  670. justify-content: end;
  671. }
  672.  
  673. .discussion-list-item .discussion-title{
  674. display: flex;
  675. flex-direction: row;
  676. flex-wrap: nowrap;
  677.  
  678. }
  679. a.discussion-list-item-report-comment[class] {
  680. all: reset;
  681. position: relative;
  682. margin: 0 0 0 0;
  683. background: inherit;
  684. color: inherit;
  685. border: 0;
  686. opacity: 0.8;
  687. text-decoration: none;
  688. font-size: 100%;
  689. }
  690. a.discussion-list-item-report-comment[class]:hover {
  691. opacity: 1.0;
  692. text-decoration: underline;
  693. }
  694.  
  695. .discussion-meta-item-script-name + .discussion-meta-item {
  696. display: inline-flex;
  697. flex-direction: row;
  698. gap: 4px;
  699. align-items: center;
  700. justify-content: flex-start;
  701. justify-items: center;
  702. }
  703.  
  704. li[data-script-id] .install-link[class] {
  705. border-radius: 0;
  706. opacity: 0.8;
  707. cursor: pointer;
  708. display: inline-flex;
  709. white-space: nowrap;
  710. position: relative;
  711. z-index: 99;
  712. }
  713.  
  714. li[data-script-id] .install-link[class]:hover {
  715. opacity: 1.0;
  716. cursor: pointer;
  717. display: inline-flex;
  718. white-space: nowrap;
  719. }
  720.  
  721. .discussion-list-item span.discussion-snippet[class] {
  722. text-overflow: ellipsis;
  723. overflow: hidden;
  724. }
  725.  
  726. div#script-list-cd[id]{
  727. /* all: revert; */
  728. padding: initial;
  729. width: initial;
  730. margin: initial;
  731. }
  732.  
  733. `
  734.  
  735. const window = {};
  736.  
  737. /** @param {typeof WinComm.createInstance} createInstance */
  738. function contentScriptText(shObject, createInstance) {
  739.  
  740. // avoid setupEthicalAdsFallback looping
  741. if (typeof window.ethicalads === "undefined") {
  742. const p = Promise.resolve([]);
  743. window.ethicalads = { wait: p };
  744. }
  745.  
  746. /*
  747. *
  748.  
  749. return new Promise((resolve, reject) => {
  750. const external = unsafeWindow.external;
  751. console.log(334, external)
  752. const scriptHandler = GM.info.scriptHandler;
  753. if (external && external.Violentmonkey && (scriptHandler || 'Violentmonkey') === 'Violentmonkey' ) {
  754. external.Violentmonkey.isInstalled(name, namespace).then((data) => resolve(data));
  755. return;
  756. }
  757.  
  758. if (external && external.Tampermonkey && (scriptHandler || 'Tampermonkey') === 'Tampermonkey') {
  759. external.Tampermonkey.isInstalled(name, namespace, (data) => {
  760. (data.installed) ? resolve(data.version) : resolve();
  761. });
  762. return;
  763. }
  764.  
  765. resolve();
  766. });
  767.  
  768. */
  769.  
  770. if (document.querySelector('#greasyfork-enhance-basic')) {
  771.  
  772.  
  773.  
  774. const setScriptOnDisabled = async (style) => {
  775.  
  776. try {
  777. const pd = Object.getOwnPropertyDescriptor(style.constructor.prototype, 'disabled');
  778. const { get, set } = pd;
  779. Object.defineProperty(style, 'disabled', {
  780. get() {
  781. return get.call(this);
  782. },
  783. set(nv) {
  784. let r = set.call(this, nv);
  785. Promise.resolve().then(chHead);
  786. return r;
  787. }
  788. })
  789. } catch (e) {
  790.  
  791. }
  792. };
  793.  
  794. document.addEventListener('style-s48', function (evt) {
  795. const target = (evt || 0).target || 0;
  796. if (!target) return;
  797. setScriptOnDisabled(target)
  798.  
  799. }, true);
  800.  
  801.  
  802. const isScriptEnabled = (style) => {
  803.  
  804. if (style instanceof HTMLStyleElement) {
  805. if (!style.hasAttribute('s48')) {
  806. style.setAttribute('s48', '');
  807. style.dispatchEvent(new CustomEvent('style-s48'));
  808. // setScriptOnDisabled(style);
  809. }
  810. return style.disabled !== true;
  811. }
  812. return false;
  813. }
  814. const chHead = () => {
  815. let p = [];
  816. if (isScriptEnabled(document.getElementById('greasyfork-enhance-basic')))
  817. p.push('basic');
  818. if (isScriptEnabled(document.getElementById('greasyfork-enhance-flat-layout')))
  819. p.push('flat-layout');
  820. if (isScriptEnabled(document.getElementById('greasyfork-enhance-animation')))
  821. p.push('animation');
  822. if (p.length >= 1)
  823. document.documentElement.setAttribute('greasyfork-enhance-k37', `|${p.join('|')}|`);
  824. else
  825. document.documentElement.removeAttribute('greasyfork-enhance-k37');
  826. }
  827. const moHead = new MutationObserver(chHead);
  828. moHead.observe(document.head, { subtree: false, childList: true });
  829. chHead();
  830.  
  831. /*
  832. const outline = document.querySelector('aside.panel > ul.outline');
  833. if(outline) {
  834. const div = document.createElement('div');
  835. //outline.replaceWith(div);
  836. //div.appendChild(outline)
  837. }
  838. */
  839.  
  840. // Promise.resolve().then(()=>{
  841. // let outline = document.querySelector('[greasyfork-enhance-k37*="|basic|"] header + aside.panel ul.outline');
  842. // if(outline){
  843. // let aside = outline.closest('aside.panel');
  844. // let header = aside.parentNode.querySelector('header');
  845. // let p = header.getBoundingClientRect().height;
  846.  
  847. // document.body.parentNode.insertBefore(aside, document.body);
  848. // // outline.style.top='0'
  849. // p+=(parseFloat(getComputedStyle(outline).marginTop.replace('px',''))||0)
  850. // outline.style.marginTop= p.toFixed(2)+'px';
  851. // }
  852.  
  853. // })
  854.  
  855. }
  856.  
  857.  
  858.  
  859. const { scriptHandler, scriptName, scriptVersion, scriptNamespace, communicationId } = shObject;
  860.  
  861. const wincomm = createInstance(communicationId);
  862.  
  863. const external = window.external;
  864.  
  865. if (external[scriptHandler]) 1;
  866. else if (external && external.Violentmonkey && (scriptHandler || 'Violentmonkey') === 'Violentmonkey') scriptHandler = 'Violentmonkey';
  867. else if (external && external.Tampermonkey && (scriptHandler || 'Tampermonkey') === 'Tampermonkey') scriptHandler = 'Tampermonkey';
  868.  
  869. const manager = external[scriptHandler];
  870.  
  871. if (!manager) {
  872.  
  873. wincomm.send('userScriptManagerNotDetected', {
  874. code: 1
  875. });
  876. return;
  877.  
  878. }
  879.  
  880. const promiseWrap = (x) => {
  881. // bug in FireFox + Violentmonkey
  882. if (typeof (x || 0) === 'object' && typeof x.then === 'function') return x; else return Promise.resolve(x);
  883. };
  884.  
  885.  
  886. const pnIsInstalled2 = (type, scriptName, scriptNamespace) => new Promise((resolve, reject) => {
  887. const resultPr = promiseWrap(manager.isInstalled(scriptName, scriptNamespace));
  888. resultPr.then((result) => resolve({
  889. type,
  890. result: typeof result === 'string' ? { version: result } : result
  891. })).catch(reject);
  892. }).catch(console.warn);
  893.  
  894.  
  895. const pnIsInstalled3 = (type, scriptName, scriptNamespace) => new Promise((resolve, reject) => {
  896. try {
  897. manager.isInstalled(scriptName, scriptNamespace, (result) => {
  898. resolve({
  899. type,
  900. result: typeof result === 'string' ? { version: result } : result
  901. });
  902. });
  903. } catch (e) {
  904. reject(e);
  905. }
  906. }).catch(console.warn);
  907.  
  908.  
  909.  
  910. const enableScriptInstallChecker = (r) => {
  911.  
  912. const { type, result } = r;
  913. let version = result.version;
  914. // console.log(type, result, version)
  915. if (version !== scriptVersion) return;
  916.  
  917. const pnIsInstalled = type < 25 ? pnIsInstalled2 : pnIsInstalled3;
  918.  
  919. wincomm.hook('_$GreasyFork$Msg$OnScriptInstallCheck', {
  920.  
  921. 'installedVersion.req': (d, evt) => {
  922. pnIsInstalled(type, d.data.name, d.data.namespace).then((r) => {
  923. if (r && 'result' in r) {
  924. wincomm.response(evt, 'installedVersion.res', {
  925. version: r.result ? (r.result.version || '') : ''
  926. });
  927. }
  928. })
  929. }
  930.  
  931. });
  932.  
  933. wincomm.send('ready', { type });
  934.  
  935. // console.log('enableScriptInstallChecker', r)
  936.  
  937.  
  938. }
  939.  
  940. const kl = manager.isInstalled.length;
  941.  
  942. if (!(kl === 2 || kl === 3)) return;
  943. const puds = kl === 2 ? [
  944. pnIsInstalled2(21, scriptName, scriptNamespace), // scriptName is GM.info.script.name not GM.info.script.name_i18n
  945. pnIsInstalled2(20, scriptName, '')
  946. ] : [
  947. pnIsInstalled3(31, scriptName, scriptNamespace),
  948. pnIsInstalled3(30, scriptName, '')
  949. ];
  950.  
  951. Promise.all(puds).then((rs) => {
  952. const [r1, r0] = rs;
  953. if (r0 && r0.result && r0.result.version) enableScriptInstallChecker(r0); // '3.1.4'
  954. else if (r1 && r1.result && r1.result.version) enableScriptInstallChecker(r1);
  955. });
  956.  
  957.  
  958.  
  959. // console.log(327, shObject, scriptHandler);
  960.  
  961. }
  962.  
  963.  
  964.  
  965. return { fields, logo, locales, blacklist, settingsCSS, pageCSS, contentScriptText }
  966.  
  967.  
  968.  
  969. })();
  970.  
  971. const inIframeFn = isInIframe ? async () => {
  972. if (window.name) {
  973. const uo = new URL(location.href);
  974. const id38 = uo.searchParams.get('id38');
  975. if (id38 && `iframe-${id38}` === window.name) {
  976.  
  977. const p38 = uo.searchParams.get('p38');
  978. const h38 = uo.searchParams.get('h38');
  979.  
  980. if (`${p38}:` === uo.protocol && `${h38}` === uo.hostname) {
  981. window.addEventListener('message', (evt)=>{
  982. if(evt && evt.data){
  983. const {id38: id38_, msg, args, fetchId} = evt.data;
  984. if(id38_ === id38){
  985. if(msg === 'fetch' && fetchId){
  986. const [url, options] = args;
  987. if(options && options.headers){
  988. options.headers = new Headers(options.headers);
  989. }
  990. fetch(url, options).then(async (response) => {
  991. let json = null;
  992. if (response.ok === true) {
  993. try {
  994. json = await response.json();
  995. } catch (e) { }
  996. }
  997. const res = {
  998. status: response.status,
  999. url: response.url,
  1000. ok: response.ok,
  1001. json
  1002. };
  1003. evt.source.postMessage({
  1004. id38,
  1005. fetchId,
  1006. msg: 'fetchResponse',
  1007. args: [res]
  1008. }, '*')
  1009. })
  1010. }
  1011. }
  1012. }
  1013. });
  1014. top.postMessage({
  1015. id38: id38,
  1016. msg: 'ready'
  1017. }, '*');
  1018. }
  1019. }
  1020. }
  1021.  
  1022. } : () => { };
  1023.  
  1024. inIframeFn() || (async () => {
  1025.  
  1026. let rafPromise = null;
  1027.  
  1028. const getRafPromise = () => rafPromise || (rafPromise = new Promise(resolve => {
  1029. requestAnimationFrame(hRes => {
  1030. rafPromise = null;
  1031. resolve(hRes);
  1032. });
  1033. }));
  1034.  
  1035. const isVaildURL = (url) => {
  1036. if (!url || typeof url !== 'string' || url.length < 23) return;
  1037. let obj = null;
  1038. try {
  1039. obj = new URL(url);
  1040. } catch (e) {
  1041. return false;
  1042. }
  1043. if (obj && obj.host === obj.hostname && !obj.port && (obj.protocol || '').startsWith('http') && obj.pathname) {
  1044. return true;
  1045. }
  1046. return false;
  1047. };
  1048.  
  1049. const installLinkPointerDownHandler = function (e) {
  1050. if (!e || !e.isTrusted) return;
  1051. const button = e.target || this;
  1052. if (button.hasAttribute('acnmd')) return;
  1053. const href = button.href;
  1054. if (!href || !isVaildURL(href)) return;
  1055. if (/\.js[^-.\w\d\s:\/\\]*$/.test(href)) {
  1056. 0 && fetch(href, {
  1057. method: "GET",
  1058. cache: 'reload',
  1059. redirect: "follow"
  1060. }).then(() => {
  1061. console.debug('code url reloaded', href);
  1062. }).catch((e) => {
  1063. console.debug(e);
  1064. });
  1065. const m = /^(https\:\/\/(cn-greasyfork|greasyfork|sleazyfork)\.org\/[_-\w\/]*scripts\/(\d+)[-\w%]*)(\/|$)/.exec(location.href)
  1066. if (m && m[1]) {
  1067. const href = `${m[1]}/code`
  1068. 0 && fetch(href, {
  1069. method: "GET",
  1070. cache: 'reload',
  1071. redirect: "follow"
  1072. }).then(() => {
  1073. console.debug('code url reloaded', href);
  1074. }).catch((e) => {
  1075. console.debug(e);
  1076. });
  1077. }
  1078.  
  1079. if (m && m[3] && href.includes('.user.js')) {
  1080. const href = `https://${location.hostname}/scripts/${m[3]}-fetching/code/${crypto.randomUUID()}.user.js?version_=${Date.now()}`
  1081. 0 && fetch(href, {
  1082. method: "GET",
  1083. cache: 'reload',
  1084. redirect: "follow"
  1085. }).then(() => {
  1086. console.debug('code url reloaded', href);
  1087. }).catch((e) => {
  1088. console.debug(e);
  1089. });
  1090. }
  1091.  
  1092.  
  1093. }
  1094.  
  1095. button.setAttribute('acnmd', '');
  1096. };
  1097.  
  1098. const setupInstallLink = (button) => {
  1099. if (!button || button.className !== 'install-link' || button.nodeName !== "A" || !button.href) return button;
  1100. button.addEventListener('pointerdown', installLinkPointerDownHandler);
  1101. return button;
  1102. };
  1103.  
  1104. function fixValue(key, def, test) {
  1105. return GM.getValue(key, def).then((v) => test(v) || GM.deleteValue(key))
  1106. }
  1107.  
  1108. const isNaNx = Number.isNaN;
  1109.  
  1110. function numberArr(arrVal) {
  1111. if (!arrVal || typeof arrVal.length !== 'number') return [];
  1112. return arrVal.filter(e => typeof e === 'number' && !isNaNx(e))
  1113. }
  1114.  
  1115. const isScriptFirstUse = await GM.getValue('firstUse', true);
  1116. await Promise.all([
  1117. fixValue('hiddenList', [], v => v && typeof v === 'object' && typeof v.length === 'number' && (v.length === 0 || typeof v[0] === 'number')),
  1118. fixValue('lastMilestone', 0, v => v && typeof v === 'number' && v >= 0)
  1119. ])
  1120.  
  1121. function createRE(t, ...opt) {
  1122. try {
  1123. return new RegExp(t, ...opt);
  1124. } catch (e) { }
  1125. return null;
  1126. }
  1127.  
  1128. const ruleFn = function (text) {
  1129. /** @type {String[]} */
  1130. const { rules, regExpArr } = this;
  1131. let text0 = text.replace(/\uE084/g, '\uE084x');
  1132. let j = 0;
  1133. for (const rule of rules) {
  1134. let r = false;
  1135. if (!rule.includes('\uE084')) {
  1136. r = (text.toLocaleLowerCase("en-US").includes(rule.toLocaleLowerCase("en-US")));
  1137. } else {
  1138. const s = rule.split(/\uE084(\d+)r/);
  1139. r = s.every((t, i) => {
  1140. if (t === undefined || t.length === 0) return true;
  1141. if (i % 2) {
  1142. return regExpArr[+t].test(text0);
  1143. } else {
  1144. return text0.includes(t.trim());
  1145. }
  1146. });
  1147. }
  1148. if (r) return j;
  1149. j++;
  1150. }
  1151. }
  1152.  
  1153. /** @param {String} txtRule */
  1154. const preprocessRule = (txtRule) => {
  1155. const regExpArr = [];
  1156. txtRule = txtRule.replace(/\uE084/g, '\uE084x');
  1157. let maxCount = 800; // avoid deadloop
  1158. while (maxCount--) {
  1159. const idx1 = txtRule.search(/\bre\//);
  1160. if (idx1 < 0) break;
  1161. const str = txtRule.substring(idx1 + 3);
  1162. let idx2 = -1;
  1163. const searcher = /(.?)\//g;
  1164. let m;
  1165. while (m = searcher.exec(str)) {
  1166. if (m[1] === '\\') continue;
  1167. idx2 = searcher.lastIndex + idx1 + 3;
  1168. break;
  1169. }
  1170. if (idx2 < 0) break;
  1171. const optionStr = txtRule.substring(idx2);
  1172. const optionM = /^[a-z]+/.exec(optionStr);
  1173. const option = optionM ? optionM[0] : '';
  1174. const regexContent = txtRule.substring(idx1 + 2 + 1, idx2 - 1);
  1175. txtRule = `${txtRule.substring(0, idx1)}${('\uE084' + regExpArr.length + 'r')}${txtRule.substring(idx2 + option.length)}`;
  1176. regExpArr.push(new RegExp(regexContent, option));
  1177. }
  1178. const rules = txtRule.split(',').map(e => e.trim());
  1179. return ruleFn.bind({ rules, regExpArr });
  1180. }
  1181.  
  1182. const useHashedScriptName = true;
  1183. const fixLibraryScriptCodeLink = true;
  1184. const addAdditionInfoLengthHint = true;
  1185.  
  1186. const id = 'greasyfork-plus';
  1187. const title = `${GM.info.script.name} v${GM.info.script.version} Settings`;
  1188. const fields = mWindow.fields;
  1189. const logo = mWindow.logo;
  1190. const nonLatins = /[^\p{Script=Latin}\p{Script=Common}\p{Script=Inherited}]/gu;
  1191. const blacklist = createRE((mWindow.blacklist || []).filter(e => !!e).join('|'), 'giu');
  1192. const hiddenList = numberArr(await GM.getValue('hiddenList', []));
  1193. const lang = document.documentElement.lang;
  1194. const locales = mWindow.locales;
  1195.  
  1196. const _isBlackList = (text) => {
  1197. if (!text || typeof text !== 'string') return false;
  1198. if (text.includes('hack') && (text.includes('EXPERIMENT_FLAGS') || text.includes('yt.'))) return false;
  1199. return blacklist.test(text);
  1200. }
  1201. const isBlackList = (name, description) => {
  1202. // To be reviewed
  1203. if (!blacklist) return false;
  1204. return _isBlackList(name) || _isBlackList(description);
  1205. }
  1206.  
  1207. function hiddenListStrToArr(str) {
  1208. if (!str || typeof str !== 'string') str = '';
  1209. return [...new Set(str ? numberArr(str.split(',').map(e => parseInt(e))) : [])];
  1210. }
  1211.  
  1212. const gmc = new GM_config({
  1213. id,
  1214. title,
  1215. fields,
  1216. css: mWindow.settingsCSS,
  1217. events: {
  1218. init: () => {
  1219. gmc.initializedResolve && gmc.initializedResolve();
  1220. gmc.initializedResolve = null;
  1221.  
  1222. },
  1223. /** @param {Document} document */
  1224. open: async (document) => {
  1225. const textarea = document.querySelector(`#${id}_field_hiddenList`);
  1226.  
  1227. const hiddenSet = new Set(numberArr(await GM.getValue('hiddenList', [])));
  1228. if (hiddenSet.size !== 0) {
  1229. const unsavedHiddenList = hiddenListStrToArr(gmc.get('hiddenList'));
  1230. const unsavedHiddenSet = new Set(unsavedHiddenList);
  1231.  
  1232. const hasDifferentItems = [...hiddenSet].some(item => !unsavedHiddenSet.has(item)) || [...unsavedHiddenSet].some(item => !hiddenSet.has(item));
  1233.  
  1234. if (hasDifferentItems) {
  1235.  
  1236. gmc.fields.hiddenList.value = [...hiddenSet].sort((a, b) => a - b).join(', ');
  1237.  
  1238. gmc.close();
  1239. gmc.open();
  1240.  
  1241. }
  1242.  
  1243.  
  1244. }
  1245.  
  1246. const resize = (target) => {
  1247. target.style.height = '';
  1248. target.style.height = `${target.scrollHeight}px`;
  1249. };
  1250.  
  1251. if (textarea) {
  1252. resize(textarea);
  1253. textarea.addEventListener('input', (event) => resize(event.target));
  1254.  
  1255. }
  1256.  
  1257. document.body.addEventListener('mousedown', (event) => {
  1258. if (event.detail > 1 && !event.ctrlKey && !event.altKey && !event.metaKey && !event.shiftKey && !event.defaultPrevented) {
  1259. event.preventDefault();
  1260. event.stopPropagation();
  1261. event.stopImmediatePropagation();
  1262. }
  1263. }, true);
  1264. },
  1265. save: async (forgotten) => {
  1266.  
  1267. if (gmc.isOpen) {
  1268. await GM.setValue('hiddenList', hiddenListStrToArr(forgotten.hiddenList));
  1269.  
  1270. UU.alert('settings saved');
  1271. gmc.close();
  1272. setTimeout(() => window.location.reload(false), 500);
  1273. }
  1274. }
  1275. }
  1276. });
  1277. gmc.initialized = new Promise(r => (gmc.initializedResolve = r));
  1278. await gmc.initialized.then();
  1279. const customBlacklistRF = preprocessRule(gmc.get('customBlacklist') || '');
  1280.  
  1281. const valHideRecentUsersWithin_ = Math.floor(+gmc.get('hideRecentUsersWithin'));
  1282. const valHideRecentUsersWithin = valHideRecentUsersWithin_ > 168 ? 168 : valHideRecentUsersWithin_ > 0 ? valHideRecentUsersWithin_ : 0;
  1283.  
  1284.  
  1285.  
  1286. /**
  1287. *
  1288. * Inserts element into the sorted array arr while maintaining order based on a comparator.
  1289. * Uses binary search to find the insertion point and then splices the element into the array.
  1290. *
  1291. * @param {Array} arr - The sorted array. (ascending order)
  1292. * @param {number} value - The number to compare.
  1293. * @param {Function} keyFn - Obtain the comparable value of the element.
  1294. */
  1295. function binarySearchLeft(arr, value, keyFn) {
  1296. let left = 0;
  1297. let right = arr.length;
  1298. while (left < right) {
  1299. const mid = Math.floor((left + right) / 2);
  1300. if (keyFn(arr[mid]) < value) {
  1301. left = mid + 1;
  1302. } else {
  1303. right = mid;
  1304. }
  1305. }
  1306. return left;
  1307. }
  1308.  
  1309. /**
  1310. * Finds the smallest index i such that arr[i][1] >= targetTime.
  1311. * Used to locate the first user in userCreations whose creation time is recent enough.
  1312. *
  1313. * @param {Array} arr - The sorted array. (ascending order)
  1314. * @param {number} targetTime - targetTime
  1315. */
  1316. function findFirstIndex(arr, targetTime) {
  1317. return binarySearchLeft(arr, targetTime, e => e[1]);
  1318. }
  1319.  
  1320.  
  1321. /**
  1322. * Finds the insertion point for element in arr to maintain sorted order.
  1323. * Used to find the range of uncertain requests in networkRequestsRCTake.
  1324. *
  1325. * @param {Array} arr - The sorted array. (ascending order)
  1326. * @param {*} element - The element to be inserted.
  1327. * @param {Function} keyFn - Obtain the comparable value of the element.
  1328. */
  1329. function insertSorted(arr, element, keyFn) {
  1330. const idx = binarySearchLeft(arr, keyFn(element), keyFn);
  1331. arr.splice(idx, 0, element);
  1332. return arr;
  1333. }
  1334.  
  1335. // Assume targetHiddenRecentDateTime is set as Date.now() - valHideRecentUsersWithin * 3600000
  1336. let targetHiddenRecentDateTime = 0;
  1337. let userCreations = [];// [userId, creationTime] sorted by creationTime
  1338. let networkRequestsRC = [];// [userId, processFn, result] sorted by userId
  1339. let recentUserMP = Promise.resolve(0);
  1340. const fetchUserCreations = () => {
  1341. if (sessionStorage.__TMP_userCreations682__) {
  1342. try {
  1343. return JSON.parse(sessionStorage.__TMP_userCreations682__);
  1344. // console.log(388, userCreations);
  1345. } catch (e) {
  1346. console.warn(e);
  1347. }
  1348. }
  1349. return [];
  1350. }
  1351. userCreations = fetchUserCreations();
  1352. // Clean up userCreations: merge with sessionStorage and trim
  1353. const cleanupUserCreations = () => {
  1354.  
  1355. // Merge with sessionStorage data
  1356. // in case the record in sessionStorage is modified by other instances as well.
  1357. const stored = fetchUserCreations();
  1358. const currentSet = new Set(userCreations.map(e => e.join(',')));
  1359. const missing = stored.filter(e => !currentSet.has(e.join(',')));
  1360. for (const element of missing) {
  1361. insertSorted(userCreations, element, e => e[1]);
  1362. }
  1363.  
  1364. // Remove redundant old entries
  1365. // since targetHiddenRecentDateTime is expected monotonic increasing, small values are useless in checking.
  1366. let deleteCount = 0;
  1367. for (let i = 0; i < userCreations.length - 1; i++) {
  1368. if (userCreations[i][1] < targetHiddenRecentDateTime && userCreations[i + 1][1] < targetHiddenRecentDateTime) {
  1369. deleteCount++;
  1370. } else {
  1371. break;
  1372. }
  1373. }
  1374. if (deleteCount > 0) {
  1375. deleteCount === 1 ? userCreations.shift() : userCreations.splice(0, deleteCount);
  1376. }
  1377.  
  1378. // Trim to max 16 elements, keeping boundary-relevant entries
  1379. while (userCreations.length > 16) {
  1380. const leftIdx = 1;
  1381. const rightIdx = userCreations.length - 2;
  1382. userCreations = userCreations.filter((e, idx) => ((idx <= leftIdx) || (idx >= rightIdx) || ((idx % 2) === 1)));
  1383. }
  1384.  
  1385. sessionStorage.__TMP_userCreations682__ = JSON.stringify(userCreations);
  1386.  
  1387. };
  1388.  
  1389. // Test if a user is recent using cached data
  1390. const testByUserCreations = (userId, targetTime)=>{
  1391. const idxJ = findFirstIndex(userCreations, targetTime);
  1392. let newFrom = Infinity, oldFrom = 0;
  1393. if (idxJ < userCreations.length) {
  1394. newFrom = userCreations[idxJ][0];
  1395. if (userId >= newFrom) return true; // User is recent
  1396. }
  1397. if (idxJ > 0) {
  1398. oldFrom = userCreations[idxJ - 1][0];
  1399. if (userId <= oldFrom) return false; // User is not recent
  1400. }
  1401. return { newFrom, oldFrom }; // Uncertain, need network request
  1402. }
  1403. // Select the next network request from the uncertain range
  1404. /** @returns {Promise | null} */
  1405. function networkRequestsRCTake() {
  1406. if (networkRequestsRC.length === 0) return null;
  1407.  
  1408. let oldFrom = 0;
  1409. let newFrom = Infinity;
  1410. if (userCreations.length > 0) {
  1411. const idx = findFirstIndex(userCreations, targetHiddenRecentDateTime);
  1412. if (idx < userCreations.length) newFrom = userCreations[idx][0];
  1413. if (idx > 0) oldFrom = userCreations[idx - 1][0];
  1414. }
  1415.  
  1416. // Find range of requests in uncertain zone (oldFrom < userId < newFrom)
  1417.  
  1418. const left = binarySearchLeft(networkRequestsRC, oldFrom + 1, e => e[0]);
  1419. // Prioritize certain not recent requests (at the beginning)
  1420. if (left > 0) {
  1421. return networkRequestsRC.shift(); // Take the first request (userId <= oldFrom)
  1422. }
  1423.  
  1424. const right = binarySearchLeft(networkRequestsRC, newFrom, e => e[0]);
  1425. // Prioritize certain recent requests (at the end)
  1426. if (right < networkRequestsRC.length) {
  1427. return networkRequestsRC.pop(); // Take the last request (userId >= newFrom)
  1428. }
  1429.  
  1430. // No certain requests left, process an uncertain one
  1431. // The entire remaining array is uncertain (left == 0, right == length)
  1432. const midIdx = Math.floor(networkRequestsRC.length / 2);
  1433. return networkRequestsRC.splice(midIdx, 1)[0];
  1434.  
  1435. }
  1436.  
  1437. // Main function to check if a user is recent
  1438. function determineRecentUserAsync(userId) {
  1439. return new Promise(resolve => {
  1440. // Check cache first
  1441. const initialCheck = testByUserCreations(userId, targetHiddenRecentDateTime);
  1442. if (typeof initialCheck === 'boolean') return resolve(initialCheck);
  1443.  
  1444. // Schedule network request
  1445. const processAsyncFn = async () => {
  1446. const check = testByUserCreations(userId, targetHiddenRecentDateTime);
  1447. // console.log('processAsyncFn', userId, targetHiddenRecentDateTime, check)
  1448. if (typeof check === 'boolean') return resolve(check);
  1449. // console.log('network request', userId)
  1450. const userData = await getUserData(userId, false); // Assume this exists
  1451. if (userData.id !== userId) return resolve(false);
  1452. const creationTime = +new Date(userData.created_at);
  1453. insertSorted(userCreations, [userId, creationTime], e => e[1]);
  1454. resolve(creationTime >= targetHiddenRecentDateTime);
  1455. cleanupUserCreations();
  1456. };
  1457.  
  1458. const request = [userId, processAsyncFn, null];
  1459. insertSorted(networkRequestsRC, request, e => e[0]);
  1460.  
  1461. // Process requests sequentially
  1462. recentUserMP = recentUserMP.then(async () => {
  1463. const entity = networkRequestsRCTake();
  1464. if (entity) await entity[1]();
  1465. });
  1466. });
  1467. }
  1468.  
  1469. if (typeof GM.registerMenuCommand === 'function') {
  1470. GM.registerMenuCommand('Configure', () => gmc.open());
  1471. GM.registerMenuCommand('Reset Everything', () => {
  1472. Promise.all([
  1473. GM.deleteValue('hiddenList'),
  1474. GM.deleteValue('lastMilestone'),
  1475. GM.deleteValue('firstUse')
  1476. ]).then(() => {
  1477. setTimeout(() => window.location.reload(false), 50);
  1478. })
  1479. });
  1480. }
  1481.  
  1482. UU.init({ id, logging: gmc.get('logging') });
  1483. UU.log(nonLatins);
  1484. UU.log(blacklist);
  1485. UU.log(hiddenList);
  1486.  
  1487. const _VM = (typeof VM !== 'undefined' ? VM : null) || {
  1488. shortcut: {
  1489. register: () => { }
  1490. }
  1491. };
  1492.  
  1493. const isGPUAccelerationAvailable = (() => {
  1494. // https://gist.github.com/cvan/042b2448fcecefafbb6a91469484cdf8
  1495. try {
  1496. const canvas = document.createElement('canvas');
  1497. return !!(canvas.getContext('webgl') || canvas.getContext('experimental-webgl'));
  1498. } catch (e) {
  1499. return false;
  1500. }
  1501. })();
  1502.  
  1503. const runLater = isGPUAccelerationAvailable ? (f) => {
  1504. requestAnimationFrame(f);
  1505. } : (f) => {
  1506. setTimeout(f, 100);
  1507. };
  1508.  
  1509. const mutationRunner = (gn, elm, options) => {
  1510. let rid = 0;
  1511. (new MutationObserver((entries) => {
  1512. if (entries && entries.length >= 1) {
  1513. const tid = rid = (rid & 1073741823) + 1;
  1514. runLater(() => {
  1515. if (tid === rid) gn();
  1516. });
  1517. }
  1518. })).observe(elm, options);
  1519. gn();
  1520. }
  1521.  
  1522.  
  1523. function fixLibraryCodeURL(code_url) {
  1524. if (/\/scripts\/(\d+)(\-[^\/]+)\/code\//.test(code_url)) {
  1525. code_url = code_url.replace(/\/scripts\/(\d+)(\-[^\/]+)\/code\//, '/scripts/$1/code/');
  1526. let qm = code_url.indexOf('?');
  1527. let s1 = code_url.substring(0, qm);
  1528. let s2 = code_url.substring(qm + 1);
  1529. if (qm > 0) {
  1530. code_url = `${decodeURI(s1)}?${s2}`;
  1531. }
  1532. }
  1533. return code_url;
  1534. }
  1535.  
  1536. function setClickToSelect(elm) {
  1537. elm.addEventListener('click', function () {
  1538. if (`${window.getSelection()}` === "") {
  1539. if (typeof this.select === 'function') {
  1540. this.select();
  1541. } else {
  1542. const range = document.createRange(); // Create a range object
  1543. range.selectNode(this); // Select the text within the element
  1544. const selection = window.getSelection(); // Get the selection object
  1545. selection.removeAllRanges(); // First clear any existing selections
  1546. selection.addRange(range); // Add the new range to the selection
  1547. }
  1548. }
  1549. });
  1550. elm.addEventListener('drag', function (evt) {
  1551. evt.preventDefault();
  1552. });
  1553. elm.addEventListener('drop', function (evt) {
  1554. evt.preventDefault();
  1555. });
  1556. elm.addEventListener('dragstart', function (evt) {
  1557. evt.preventDefault();
  1558. });
  1559. }
  1560.  
  1561. const copyText = typeof (((window.navigator || 0).clipboard || 0).writeText) === 'function' ? (text) => {
  1562. navigator.clipboard.writeText(text).then(function () {
  1563. //
  1564. }).catch(function (err) {
  1565. alert("Unable to Copy");
  1566. });
  1567. } : (text) => {
  1568. const textToCopy = document.createElement('strong');
  1569. textToCopy.style.position = 'fixed';
  1570. textToCopy.style.opacity = '0';
  1571. textToCopy.style.top = '-900vh';
  1572. textToCopy.textContent = text;
  1573. document.body.appendChild(textToCopy);
  1574.  
  1575. const range = document.createRange(); // Create a range object
  1576. range.selectNode(textToCopy); // Select the text within the element
  1577.  
  1578. const selection = window.getSelection(); // Get the selection object
  1579. selection.removeAllRanges(); // First clear any existing selections
  1580. selection.addRange(range); // Add the new range to the selection
  1581.  
  1582. try {
  1583. document.execCommand('copy'); // Try to copy the selected text
  1584. } catch (err) {
  1585. alert("Unable to Copy");
  1586. }
  1587.  
  1588. selection.removeAllRanges(); // Remove the selection range after copying
  1589. textToCopy.remove();
  1590. };
  1591.  
  1592.  
  1593. let avoidDuplication = 0;
  1594. const avoidDuplicationF = () => {
  1595. const p = avoidDuplication;
  1596. avoidDuplication = Date.now();
  1597. if (avoidDuplication - p < 30) return false;
  1598. return true;
  1599. }
  1600. // https://violentmonkey.github.io/vm-shortcut/
  1601. const shortcuts = [
  1602. ['ctrlcmd-alt-keys', () => avoidDuplicationF() && gmc.open()],
  1603. ['ctrlcmd-alt-keyb', () => avoidDuplicationF() && toggleListDisplayingItem('blacklisted')],
  1604. ['ctrlcmd-alt-keyh', () => avoidDuplicationF() && toggleListDisplayingItem('hidden')]
  1605. ]
  1606. for (const [scKey, scFn] of shortcuts) {
  1607. _VM.shortcut.register(scKey, scFn);
  1608. }
  1609.  
  1610. const addSettingsToMenu = () => {
  1611. const nav = document.querySelector('#site-nav > nav')
  1612. if (!nav) return;
  1613.  
  1614. const scriptName = GM.info.script.name;
  1615. const scriptVersion = GM.info.script.version;
  1616. const menu = document.createElement('li');
  1617. menu.classList.add(id);
  1618. menu.setAttribute('alt', `${scriptName} ${scriptVersion}`);
  1619. menu.setAttribute('title', `${scriptName} ${scriptVersion}`);
  1620. const link = document.createElement('a');
  1621. link.setAttribute('href', '#');
  1622. link.textContent = GM.info.script.name;
  1623. menu.appendChild(link);
  1624. nav.insertBefore(menu, document.querySelector('#site-nav > nav > li:first-of-type'));
  1625.  
  1626. menu.addEventListener('click', (e) => {
  1627. e.preventDefault();
  1628. e.stopPropagation();
  1629. e.stopImmediatePropagation();
  1630. gmc.open();
  1631. });
  1632. };
  1633.  
  1634.  
  1635. const toggleListDisplayingItem = (t) => {
  1636.  
  1637. const m = document.documentElement;
  1638.  
  1639. const p = t + '-shown';
  1640. let currentIsShown = m.hasAttribute(p)
  1641. if (!currentIsShown) {
  1642. m.setAttribute(p, '')
  1643. } else {
  1644. m.removeAttribute(p)
  1645. }
  1646.  
  1647. }
  1648.  
  1649. const createListOptionGroup = () => {
  1650.  
  1651. const html = `<div class="list-option-group" id="${id}-options">${GM.info.script.name} Lists:<ul>
  1652. <li class="list-option blacklisted"><a href="#" id="hyperlink-35389"></a></li>
  1653. <li class="list-option hidden"><a href="#" id="hyperlink-40361"></a></li>
  1654. </ul></div>`;
  1655. const firstOptionGroup = document.querySelector('.list-option-groups > div');
  1656. firstOptionGroup && firstOptionGroup.insertAdjacentHTML('beforebegin', html);
  1657.  
  1658. const blacklistedOption = document.querySelector(`#${id}-options li.blacklisted`);
  1659. blacklistedOption && blacklistedOption.addEventListener('click', (evt) => {
  1660. evt.preventDefault();
  1661. toggleListDisplayingItem('blacklisted');
  1662. }, false);
  1663.  
  1664. const hiddenOption = document.querySelector(`#${id}-options li.hidden`);
  1665. hiddenOption && hiddenOption.addEventListener('click', (evt) => {
  1666. evt.preventDefault();
  1667. toggleListDisplayingItem('hidden');
  1668. }, false);
  1669.  
  1670. }
  1671.  
  1672. const addOptions = (scriptList) => {
  1673. if (!scriptList) return;
  1674. createListOptionGroup();
  1675. mutationRunner(() => {
  1676. let aBlackList = document.querySelector('#hyperlink-35389');
  1677. let aHidden = document.querySelector('#hyperlink-40361');
  1678. if (!aBlackList || !aHidden) return;
  1679. aBlackList.textContent = `Blacklisted scripts (${document.querySelectorAll('.script-list li.blacklisted').length})`;
  1680. aHidden.textContent = `Hidden scripts (${document.querySelectorAll('.script-list li.hidden').length})`;
  1681. }, scriptList, { childList: true, subtree: true });
  1682. };
  1683.  
  1684.  
  1685. const PromiseExternal = ((resolve_, reject_) => {
  1686. const h = (resolve, reject) => { resolve_ = resolve; reject_ = reject };
  1687. return class PromiseExternal extends Promise {
  1688. constructor(cb = h) {
  1689. super(cb);
  1690. if (cb === h) {
  1691. /** @type {(value: any) => void} */
  1692. this.resolve = resolve_;
  1693. /** @type {(reason?: any) => void} */
  1694. this.reject = reject_;
  1695. }
  1696. }
  1697. };
  1698. })();
  1699.  
  1700. const corsFetchMap = new Map();
  1701.  
  1702. const corsFetch = async (url, options) => {
  1703. if (top !== window) return;
  1704. const uo = new URL(url);
  1705. const protocol = uo.protocol.replace(/[^\w]+/g, '');
  1706. const hostname = uo.hostname;
  1707. const origin0 = `${protocol}://${hostname}`;
  1708. let promiseF = null;
  1709. let prFn = corsFetchMap.get(origin0);
  1710. for (let i = 0; i < 2; i++) {
  1711. if (!prFn) {
  1712. prFn = new Promise((resolve) => {
  1713. let iframe = document.createElement('iframe');
  1714. const rid = `${Math.floor(Math.random() * 314159265359 + 314159265359).toString(36)}`;
  1715. iframe.id = `iframe-${rid}`;
  1716. iframe.name = `iframe-${rid}`;
  1717. window.addEventListener('message', (evt) => {
  1718. if (evt && evt.origin === origin0) {
  1719. const data = evt.data;
  1720. if (data && data.id38) {
  1721. const { id38, msg, fetchId: fetchId_, args } = data;
  1722. if (msg === 'ready') {
  1723. const iframeWindow = evt.source;
  1724. resolve((...args) => {
  1725. if (!iframe.isConnected) return -1;
  1726. const fetchId = `${Math.floor(Math.random() * 314159265359 + 314159265359).toString(36)}`;
  1727. const promise = new PromiseExternal();
  1728. corsFetchMap.set(`${id38}-${fetchId}`, promise);
  1729. iframeWindow.postMessage({
  1730. id38,
  1731. msg: 'fetch',
  1732. fetchId,
  1733. args
  1734. }, '*');
  1735. return promise;
  1736. });
  1737. } else if (msg === 'fetchResponse') {
  1738. const promise = corsFetchMap.get(`${id38}-${fetchId_}`);
  1739. if (promise) {
  1740. corsFetchMap.delete(`${id38}-${fetchId_}`);
  1741. promise.resolve(args[0]);
  1742. }
  1743. }
  1744. }
  1745. }
  1746. });
  1747. iframe.src = `${protocol}://${hostname}/robots.txt?id38=${rid}&p38=${protocol}&h38=${hostname}`;
  1748. Object.assign(iframe.style, {
  1749. 'position': 'fixed',
  1750. 'left': '-300px',
  1751. 'top': '-300px',
  1752. 'width': '30px',
  1753. 'height': '30px',
  1754. 'pointerEvents': 'none',
  1755. 'zIndex': '-1',
  1756. 'contain': 'strict'
  1757. });
  1758. (document.body || document.documentElement).appendChild(iframe);
  1759. });
  1760. corsFetchMap.set(origin0, prFn);
  1761. }
  1762. const fetchFn = await prFn.then();
  1763. const promise = fetchFn(url, options);
  1764. if (promise === -1) {
  1765. corsFetchMap.delete(origin0);
  1766. prFn = null;
  1767. continue;
  1768. }
  1769. if (promise && typeof promise.then === 'function') {
  1770. promiseF = promise;
  1771. break;
  1772. }
  1773. }
  1774. if (!promiseF) return null;
  1775. const promiseResult = await promiseF.then();
  1776. return promiseResult;
  1777. };
  1778.  
  1779. const standardFetch = async (url, options) => {
  1780. if (options && options.headers) {
  1781. options.headers = new Headers(options.headers);
  1782. }
  1783. const response = await fetch(url, options);
  1784. let json = null;
  1785. if (response.ok === true) {
  1786. try {
  1787. json = await response.json();
  1788. } catch (e) { }
  1789. }
  1790. const res = {
  1791. status: response.status,
  1792. url: response.url,
  1793. ok: response.ok,
  1794. json
  1795. };
  1796. return res;
  1797. }
  1798.  
  1799. /**
  1800. * Get script data from Greasy Fork镜像 API
  1801. *
  1802. * @param {number} id Script ID
  1803. * @returns {Promise} Script data
  1804. */
  1805. let networkMP1 = Promise.resolve();
  1806. let networkMP2 = Promise.resolve();
  1807. let previousIsCache = false;
  1808. // let ss = [];
  1809. // var sum = function(nums) {
  1810. // var total = 0;
  1811. // for (var i = 0, len = nums.length; i < len; i++) total += nums[i];
  1812. // return total;
  1813. // };
  1814. let reqStoresA = new Map();
  1815. let reqStoresB = new Map();
  1816.  
  1817. const getOldestEntry = (noCache)=>{
  1818. const reqStores = noCache ? reqStoresB : reqStoresA;
  1819. const oldestEntry = reqStores.entries().next();
  1820. if(!oldestEntry || !oldestEntry.value) return [];
  1821. const id = oldestEntry.value[0]
  1822. const req = oldestEntry.value[1]
  1823. reqStores.delete(id);
  1824. return [id, req];
  1825. }
  1826.  
  1827. let mutexC = Promise.resolve();
  1828. const getScriptDataAN = (noCache)=>{
  1829.  
  1830. mutexC = mutexC.then(async () => {
  1831.  
  1832. const [id, req] = getOldestEntry(noCache);
  1833.  
  1834. if (!(id > 0)) return;
  1835.  
  1836. const DO_CORS = /^(cn-greasyfork|greasyfork|sleazyfork)\.org$/.test(window.location.hostname) ? `api.${window.location.hostname}` : '';
  1837. const url = `https://${DO_CORS || window.location.hostname}/scripts/${id}.json`;
  1838. const fetchUrl = sessionStorage.getItem(`redirect41-${url}`) || url;
  1839.  
  1840. const onPageElement = document.querySelector(`[data-script-namespace][data-script-id="${id || 'null'}"][data-script-name][data-script-version][href]`)
  1841. if (onPageElement && /^https\:\/\/update\.\w+\.org\/scripts\/\d+\/[^.?\/]+\.user\.js$/.test(onPageElement.getAttribute('href') || '')) {
  1842. const result = {
  1843. "id": +onPageElement.getAttribute('data-script-id'),
  1844. // "created_at": "2023-08-24T21:16:50.000Z",
  1845. // "daily_installs": 21,
  1846. // "total_installs": 3310,
  1847. // "code_updated_at": "2023-12-20T07:46:54.000Z",
  1848. // "support_url": null,
  1849. // "fan_score": "74.1",
  1850. "namespace": `${onPageElement.getAttribute('data-script-namespace')}`,
  1851. // "contribution_url": null,
  1852. // "contribution_amount": null,
  1853. // "good_ratings": 11,
  1854. // "ok_ratings": 0,
  1855. // "bad_ratings": 0,
  1856. // "users": [
  1857. // {
  1858. // "id": 371179,
  1859. // "name": "𝖢𝖸 𝖥𝗎𝗇𝗀",
  1860. // "url": "https://gf.qytechs.cn/users/371179-%F0%9D%96%A2%F0%9D%96%B8-%F0%9D%96%A5%F0%9D%97%8E%F0%9D%97%87%F0%9D%97%80"
  1861. // }
  1862. // ],
  1863. "name": `${onPageElement.getAttribute('data-script-name')}`,
  1864. // "description": "Adds various features and improves the Greasy Fork镜像 experience",
  1865. // "url": "https://gf.qytechs.cn/scripts/473830-greasy-fork",
  1866. // "code_url": "https://update.gf.qytechs.cn/scripts/473830/Greasy%20Fork%2B%2B.user.js",
  1867. "code_url": `${onPageElement.getAttribute('href')}`,
  1868. // "license": "MIT License",
  1869. "version": `${onPageElement.getAttribute('data-script-version')}`,
  1870. // "locale": "en",
  1871. // "deleted": false
  1872. };
  1873. req.resolve(result);
  1874. return;
  1875. }
  1876. await (networkMP1 = networkMP1.then(() => new Promise(unlock => {
  1877. const maxAgeInSeconds = 900;
  1878. const rd = previousIsCache ? 1 : Math.floor(Math.random() * 80 + 80);
  1879. let fetchStart = 0;
  1880.  
  1881. const fetchOptions = noCache ? {
  1882. method: 'GET',
  1883. cache: 'reload',
  1884. credentials: 'omit',
  1885. headers: {
  1886. 'Cache-Control': `max-age=${maxAgeInSeconds}`,
  1887. }
  1888. } : {
  1889. method: 'GET',
  1890. cache: 'force-cache',
  1891. credentials: 'omit',
  1892. headers: {
  1893. 'Cache-Control': `max-age=${maxAgeInSeconds}`,
  1894. }
  1895. };
  1896.  
  1897. new Promise(r => setTimeout(r, rd))
  1898. .then(() => {
  1899. fetchStart = Date.now();
  1900. })
  1901. .then(() => DO_CORS ? corsFetch(fetchUrl, fetchOptions): standardFetch(fetchUrl, fetchOptions))
  1902. .then((response) => {
  1903.  
  1904. if (fetchUrl !== response.url) {
  1905. sessionStorage.setItem(`redirect41-${url}`, response.url);
  1906. sessionStorage.setItem(`redirect41-${fetchUrl}`, response.url);
  1907. }
  1908. let fetchStop = Date.now();
  1909. // const dd = fetchStop - fetchStart;
  1910. // dd (cache) = {min: 1, max: 8, avg: 3.7}
  1911. // dd (normal) = {min: 136, max: 316, avg: 162.62}
  1912. // ss.push(dd)
  1913. // ss.maxValue = Math.max(...ss);
  1914. // ss.minValue = Math.min(...ss);
  1915. // ss.avgValue = sum(ss)/ss.length;
  1916. // console.log(dd)
  1917. // console.log(ss)
  1918. previousIsCache = (fetchStop - fetchStart) < (3.7 + 162.62) / 2;
  1919. UU.log(`${response.status}: ${response.url}`)
  1920. // UU.log(response)
  1921. if (response.ok === true) {
  1922. unlock();
  1923. return response.json;
  1924. }
  1925. if (response.status === 503) {
  1926. return new Promise(r => setTimeout(r, 270 + rd)).then(() => {
  1927. unlock();
  1928. return getScriptData(id, true);
  1929. });
  1930. }
  1931. if (response.status === 404) {
  1932. // script XXXX has been reported and is pending review by a moderator.
  1933. unlock();
  1934. return null
  1935. }
  1936. console.warn(response.status, response);
  1937. new Promise(r => setTimeout(r, 470)).then(unlock); // reload later
  1938. })
  1939. .then((data) => req.resolve(data))
  1940. .catch((e) => {
  1941. unlock();
  1942. UU.log(id, url)
  1943. console.warn(e)
  1944. // reject(e)
  1945. })
  1946. })).catch(() => { }))
  1947.  
  1948. });
  1949.  
  1950. }
  1951. const getScriptData = (id, noCache) => {
  1952. if (!(+id > 0)) return Promise.resolve();
  1953. id = +id;
  1954. const reqStores = noCache ? reqStoresB : reqStoresA;
  1955. const cachedReq = reqStores.get(id);
  1956. if (cachedReq) return cachedReq;
  1957. const req = new PromiseExternal();
  1958. reqStores.set(id, req);
  1959. getScriptDataAN(noCache);
  1960. return req;
  1961. }
  1962.  
  1963. /**
  1964. * Get user data from Greasy Fork镜像 API
  1965. *
  1966. * @param {string} userID User ID
  1967. * @returns {Promise} User data
  1968. */
  1969. const getUserData = (userID, noCache) => {
  1970. if (!(userID >= 0)) return Promise.resolve()
  1971.  
  1972. const DO_CORS = /^(cn-greasyfork|greasyfork|sleazyfork)\.org$/.test(window.location.hostname) ? `api.${window.location.hostname}` : '';
  1973. const url = `https://${DO_CORS || window.location.hostname}/users/${userID}.json`;
  1974. const fetchUrl = sessionStorage.getItem(`redirect41-${url}`) || url;
  1975. return new Promise((resolve, reject) => {
  1976.  
  1977.  
  1978. networkMP2 = networkMP2.then(() => new Promise(unlock => {
  1979.  
  1980. const maxAgeInSeconds = 900;
  1981. const rd = Math.floor(Math.random() * 80 + 80);
  1982.  
  1983. const fetchOptions = noCache ? {
  1984. method: 'GET',
  1985. cache: 'reload',
  1986. credentials: 'omit',
  1987. headers: {
  1988. 'Cache-Control': `max-age=${maxAgeInSeconds}`,
  1989. }
  1990. } : {
  1991. method: 'GET',
  1992. cache: 'force-cache',
  1993. credentials: 'omit',
  1994. headers: {
  1995. 'Cache-Control': `max-age=${maxAgeInSeconds}`,
  1996. }
  1997. };
  1998.  
  1999. new Promise(r => setTimeout(r, rd))
  2000.  
  2001. .then(() => DO_CORS ? corsFetch(fetchUrl, fetchOptions) : standardFetch(fetchUrl, fetchOptions))
  2002. .then((response) => {
  2003.  
  2004. if (fetchUrl !== response.url) {
  2005. sessionStorage.setItem(`redirect41-${url}`, response.url);
  2006. sessionStorage.setItem(`redirect41-${fetchUrl}`, response.url);
  2007. }
  2008.  
  2009. UU.log(`${response.status}: ${response.url}`)
  2010. if (response.ok === true) {
  2011. unlock();
  2012. return response.json;
  2013. }
  2014. if (response.status === 503) {
  2015. return new Promise(r => setTimeout(r, 270 + rd)).then(() => {
  2016. unlock();
  2017. return getUserData(userID, true); // reload later
  2018. });
  2019. }
  2020. if (response.status === 404) {
  2021. // user XXXX has been reported and is pending review by a moderator. ????
  2022. unlock();
  2023. return null
  2024. }
  2025. console.warn(response.status, response);
  2026. new Promise(r => setTimeout(r, 470)).then(unlock);
  2027. })
  2028. .then((data) => resolve(data))
  2029. .catch((e) => {
  2030. setTimeout(() => {
  2031. unlock()
  2032. }, 270)
  2033. UU.log(userID, url)
  2034. console.warn(e)
  2035. // reject(e)
  2036. })
  2037.  
  2038.  
  2039.  
  2040. })).catch(() => { })
  2041.  
  2042. });
  2043. }
  2044. const getTotalInstalls = (data) => {
  2045. if (!data || !data.scripts) return;
  2046. return new Promise((resolve, reject) => {
  2047. const totalInstalls = [];
  2048.  
  2049. data.scripts.forEach((element) => {
  2050. totalInstalls.push(parseInt(element.total_installs, 10));
  2051. });
  2052.  
  2053. resolve(totalInstalls.reduce((a, b) => a + b, 0));
  2054. });
  2055. };
  2056.  
  2057.  
  2058. const communicationId = WinComm.newCommunicationId();
  2059. const wincomm = WinComm.createInstance(communicationId);
  2060.  
  2061.  
  2062. const isInstalled = (script) => {
  2063. return new Promise((resolve, reject) => {
  2064.  
  2065. promiseScriptCheck.then(d => {
  2066.  
  2067. if (!d) return null;
  2068.  
  2069. const data = d.data;
  2070. const al = data.type % 10;
  2071. if (al === 0) {
  2072. // no namespace
  2073. resolve([null, script.name, '']);
  2074. } else if (al === 1) {
  2075. // namespace
  2076.  
  2077. if (!script.namespace) {
  2078.  
  2079. getRafPromise() // foreground
  2080. .then(() => getScriptData(script.id))
  2081. .then((script) => {
  2082. resolve([null, script.name, script.namespace]);
  2083. });
  2084.  
  2085. } else {
  2086.  
  2087. resolve([null, script.name, script.namespace]);
  2088. }
  2089.  
  2090. }
  2091.  
  2092.  
  2093. })
  2094.  
  2095.  
  2096. }).then((res) => {
  2097.  
  2098.  
  2099. return new Promise((resolve, reject) => {
  2100.  
  2101. if (!res) return '';
  2102.  
  2103.  
  2104. const [_, name, namespace] = res;
  2105. wincomm.request('installedVersion.req', {
  2106. name,
  2107. namespace
  2108. }).then(d => {
  2109. resolve(d.data.version)
  2110. })
  2111.  
  2112. })
  2113.  
  2114. })
  2115.  
  2116. /*
  2117. const external = unsafeWindow.external;
  2118. const scriptHandler = GM.info.scriptHandler;
  2119. if (external && external.Violentmonkey && (scriptHandler || 'Violentmonkey') === 'Violentmonkey') {
  2120. external.Violentmonkey.isInstalled(name, namespace).then((data) => resolve(data));
  2121. return;
  2122. }
  2123.  
  2124. if (external && external.Tampermonkey && (scriptHandler || 'Tampermonkey') === 'Tampermonkey') {
  2125. external.Tampermonkey.isInstalled(name, namespace, (data) => {
  2126. (data.installed) ? resolve(data.version) : resolve();
  2127. });
  2128. return;
  2129. }
  2130. */
  2131.  
  2132.  
  2133. };
  2134.  
  2135. const compareVersions = (v1, v2) => {
  2136. if (!v1 || !v2) return NaN;
  2137. if (v1 === null || v2 === null) return NaN;
  2138. if (v1 === v2) return 0;
  2139.  
  2140. const sv1 = v1.split('.').map((index) => parseInt(index));
  2141. const sv2 = v2.split('.').map((index) => parseInt(index));
  2142.  
  2143. const count = Math.max(sv1.length, sv2.length);
  2144.  
  2145. for (let index = 0; index < count; index++) {
  2146. if (isNaNx(sv1[index]) || isNaNx(sv2[index])) return NaN;
  2147. if (sv1[index] > sv2[index]) return 1;
  2148. if (sv1[index] < sv2[index]) return -1;
  2149. }
  2150.  
  2151. return 0;
  2152. };
  2153.  
  2154.  
  2155. /**
  2156. * Return label for the hide script button
  2157. *
  2158. * @param {boolean} hidden Is hidden
  2159. * @returns {string} Label
  2160. */
  2161. const blockLabel = (hidden) => {
  2162. return hidden ? (locales[lang] ? locales[lang].notHide : locales.en.notHide) : (locales[lang] ? locales[lang].hide : locales.en.hide)
  2163. }
  2164.  
  2165. /**
  2166. * Return label for the install button
  2167. *
  2168. * @param {number} update Update value
  2169. * @returns {string} Label
  2170. */
  2171. const installLabel = (update) => {
  2172. switch (update) {
  2173. case 0: {
  2174. return locales[lang] ? locales[lang].reinstall : locales.en.reinstall
  2175. }
  2176. case 1: {
  2177. return locales[lang] ? locales[lang].update : locales.en.update
  2178. }
  2179. case -1: {
  2180. return locales[lang] ? locales[lang].downgrade : locales.en.downgrade
  2181. }
  2182. default: {
  2183. return locales[lang] ? locales[lang].install : locales.en.install
  2184. }
  2185. }
  2186. }
  2187.  
  2188. const hideBlacklistedDiscussion = (element, list) => {
  2189.  
  2190. const scriptLink = element.querySelector('a.script-link')
  2191. const m = /\/scripts\/(\d+)/.exec(scriptLink);
  2192. const id = m ? +m[1] : 0;
  2193. if (!(id > 0)) return;
  2194.  
  2195. switch (list) {
  2196. case 'hiddenList': {
  2197. const container = element.closest('.discussion-list-container') || element;
  2198. if (hiddenList.indexOf(id) >= 0) {
  2199. container.classList.add('hidden');
  2200. }
  2201. // if (customBlacklist && (customBlacklist.test(name) || customBlacklist.test(description)) && !element.classList.contains('blacklisted')) {
  2202. // element.classList.add('blacklisted', 'custom-blacklist');
  2203. // if (gmc.get('hideBlacklistedScripts') && gmc.get('debugging')) {
  2204. // let scriptLink = element.querySelector('.script-link');
  2205. // if (scriptLink) { scriptLink.textContent += ' (custom-blacklist)'; }
  2206. // }
  2207. // }
  2208. break;
  2209. }
  2210. default:
  2211. UU.log('No blacklists');
  2212. break;
  2213. }
  2214.  
  2215. }
  2216. const hideBlacklistedScript = (element, list) => {
  2217. if (!element) return;
  2218. const scriptLink = element.querySelector('.script-link')
  2219.  
  2220. const name = scriptLink ? scriptLink.textContent : '';
  2221. const descriptionElem = element.querySelector('.script-description')
  2222. const description = descriptionElem ? descriptionElem.textContent : '';
  2223.  
  2224. if (!name) return;
  2225.  
  2226. switch (list) {
  2227. case 'nonLatins':
  2228. if ((nonLatins.test(name) || nonLatins.test(description)) && !element.classList.contains('blacklisted')) {
  2229. element.classList.add('blacklisted', 'non-latins');
  2230. if (gmc.get('hideBlacklistedScripts') && gmc.get('debugging')) {
  2231. let scriptLink = element.querySelector('.script-link');
  2232. if (scriptLink) { scriptLink.textContent += ' (non-latin)'; }
  2233. }
  2234. }
  2235. break;
  2236. case 'blacklist':
  2237. if (isBlackList(name, description) && !element.classList.contains('blacklisted')) {
  2238. element.classList.add('blacklisted', 'blacklist');
  2239. if (gmc.get('hideBlacklistedScripts') && gmc.get('debugging')) {
  2240. let scriptLink = element.querySelector('.script-link');
  2241. if (scriptLink) { scriptLink.textContent += ' (blacklist)'; }
  2242. }
  2243. }
  2244. break;
  2245. case 'customBlacklist': {
  2246. const customBlacklist = customBlacklistRF;
  2247. if (customBlacklist && (customBlacklist(name) >= 0 || customBlacklist(description) >= 0) && !element.classList.contains('blacklisted')) {
  2248. element.classList.add('blacklisted', 'custom-blacklist');
  2249. if (gmc.get('hideBlacklistedScripts') && gmc.get('debugging')) {
  2250. let scriptLink = element.querySelector('.script-link');
  2251. if (scriptLink) { scriptLink.textContent += ' (custom-blacklist)'; }
  2252. }
  2253. }
  2254. break;
  2255. }
  2256. default:
  2257. UU.log('No blacklists');
  2258. break;
  2259. }
  2260. };
  2261.  
  2262. const hideHiddenScript = (element, id, list) => {
  2263. id = +id;
  2264. if (!(id >= 0)) return;
  2265.  
  2266. const isInHiddenList = () => hiddenList.indexOf(id) !== -1;
  2267. const updateScriptLink = (shouldHide) => {
  2268. if (gmc.get('hideHiddenScript') && gmc.get('debugging')) {
  2269. let scriptLink = element.querySelector('.script-link');
  2270. if (scriptLink) {
  2271. if (shouldHide) {
  2272. scriptLink.innerHTML += ' (hidden)';
  2273. } else {
  2274. scriptLink.innerHTML = scriptLink.innerHTML.replace(' (hidden)', '');
  2275. }
  2276. }
  2277. }
  2278. };
  2279.  
  2280. // Check for initial state and set it
  2281. if (isInHiddenList()) {
  2282. element.classList.add('hidden');
  2283. updateScriptLink(true);
  2284. }
  2285.  
  2286. // Add button to hide the script
  2287. const insertButtonHTML = (selector, html) => {
  2288. const target = element.querySelector(selector);
  2289. if (!target) return;
  2290. let p = document.createElement('template');
  2291. p.innerHTML = html;
  2292. target.parentNode.insertBefore(p.content.firstChild, target.nextSibling);
  2293. };
  2294.  
  2295. const isHidden = element.classList.contains('hidden');
  2296. const blockButtonHTML = `<span class=block-button role=button style-16377>${blockLabel(isHidden)}</span>`;
  2297. const blockButtonHeaderHTML = `<span class=block-button role=button style-77329 style="">${blockLabel(isHidden)}</span>`;
  2298.  
  2299. insertButtonHTML('.badge-js, .badge-css', blockButtonHTML);
  2300. insertButtonHTML('header h2', blockButtonHeaderHTML);
  2301.  
  2302. // Add event listener
  2303. const button = element.querySelector('.block-button');
  2304. if (button) {
  2305. button.addEventListener('click', (event) => {
  2306. event.stopPropagation();
  2307. event.stopImmediatePropagation();
  2308.  
  2309. if (!isInHiddenList()) {
  2310. hiddenList.push(id);
  2311. GM.setValue('hiddenList', hiddenList);
  2312.  
  2313. element.classList.add('hidden');
  2314. updateScriptLink(true);
  2315.  
  2316. } else {
  2317. const index = hiddenList.indexOf(id);
  2318. hiddenList.splice(index, 1);
  2319. GM.setValue('hiddenList', hiddenList);
  2320.  
  2321. element.classList.remove('hidden');
  2322. updateScriptLink(false);
  2323. }
  2324.  
  2325. const blockBtn = element.querySelector('.block-button');
  2326. if (blockBtn) blockBtn.textContent = blockLabel(element.classList.contains('hidden'));
  2327. });
  2328. }
  2329. };
  2330.  
  2331. const insertButtonHTML = (element, selector, html) => {
  2332. const target = element.querySelector(selector);
  2333. if (!target) return;
  2334. let p = document.createElement('template');
  2335. p.innerHTML = html;
  2336. let button = p.content.firstChild
  2337. target.parentNode.insertBefore(button, target.nextSibling);
  2338. return button;
  2339. };
  2340.  
  2341. const addInstallButton = (element, url) => {
  2342. return setupInstallLink(insertButtonHTML(element, '.badge-js, .badge-css', `<a class="install-link" href="${url}" style-54998></a>`));
  2343. };
  2344.  
  2345. async function digestMessage(message, algo) {
  2346. const encoder = new TextEncoder();
  2347. const data = encoder.encode(message);
  2348. const hash = await crypto.subtle.digest(algo, data);
  2349. return hash;
  2350. }
  2351.  
  2352. function qexString(buffer) {
  2353. const byteArray = new Uint8Array(buffer);
  2354. const len = byteArray.length;
  2355. const hexCodes = new Array(len * 2);
  2356. const chars = 'a4b3c5d7e6f9h2t';
  2357. for (let i = 0, j = 0; i < len; i++) {
  2358. const byte = byteArray[i];
  2359. hexCodes[j++] = chars[byte >> 4];
  2360. hexCodes[j++] = chars[byte & 0x0F];
  2361. };
  2362. return hexCodes.join('');
  2363. }
  2364.  
  2365. const encodeFileName = (s) => {
  2366. if (!s || typeof s !== 'string') return s;
  2367. s = s.replace(/[.!~*'"();\/\\?@&=$,#]/g, '-').replace(/\s+/g, ' ');
  2368. return encodeURI(s);
  2369. }
  2370.  
  2371. const isLibraryURLWithVersion = (url) => {
  2372. if (!url || typeof url !== 'string') return;
  2373.  
  2374. if (url.includes('.js?version=')) return true;
  2375.  
  2376. if (/\/scripts\/\d+\/\d+\/[^.!~*'"();\/\\?@&=$,#]+\.js/.test(url)) return true;
  2377. return false;
  2378.  
  2379. }
  2380.  
  2381. const showInstallButton = async (scriptID, element) => {
  2382.  
  2383. await getRafPromise().then();
  2384. // if(document.querySelector(`li[data-script-id="${scriptID}"]`))
  2385. let _baseScript = null;
  2386. if (element.nodeName === 'LI' && element.hasAttribute('data-script-id') && element.getAttribute('data-script-id') === `${scriptID}` && element.getAttribute('data-script-language') === 'js') {
  2387.  
  2388. const version = element.getAttribute('data-script-version') || ''
  2389.  
  2390. let scriptCodeURL = element.getAttribute('data-code-url');
  2391. if (!scriptCodeURL || !isVaildURL(scriptCodeURL)) {
  2392.  
  2393. const name = element.getAttribute('data-script-name') || ''
  2394. // if (!/[^\x00-\x7F]/.test(name)) {
  2395.  
  2396. // const scriptName = useHashedScriptName ? qexString(await digestMessage(`${+scriptID} ${version}`, 'SHA-1')).substring(0, 8) : encodeURI(name);
  2397. // const token = useHashedScriptName ? `${scriptName.substring(0, 2)}${scriptName.substring(scriptName.length - 2, scriptName.length)}` : String.fromCharCode(Date.now() % 26 + 97) + Math.floor(Math.random() * 19861 + 19861).toString(36);
  2398. const scriptFilename = element.getAttribute('data-script-type') === 'library' ? `${encodeFileName(name)}.js` : `${encodeFileName(name)}.user.js`;
  2399. // const scriptFilename = `${scriptName}.user.js`;
  2400.  
  2401. // code_url: `https://${location.hostname}/scripts/${scriptID}-${token}/code/${scriptFilename}`,
  2402. // code_url: `https://update.${location.hostname}/scripts/${scriptID}.user.js`,
  2403. scriptCodeURL = `https://update.${location.hostname}/scripts/${scriptID}/${scriptFilename}`
  2404. }
  2405. _baseScript = {
  2406. id: +scriptID,
  2407. // name: name,
  2408. code_url: scriptCodeURL,
  2409. version: version
  2410. }
  2411. // }
  2412.  
  2413. }
  2414.  
  2415. const baseScript = _baseScript || (await getScriptData(scriptID));
  2416.  
  2417. if ((element.nodeName === 'LI' && element.getAttribute('data-script-type') === 'library') || (baseScript.code_url.includes('.js?version='))) {
  2418.  
  2419. let scriptCodeURL = element.getAttribute('data-code-url');
  2420.  
  2421. if (!scriptCodeURL || !isVaildURL(scriptCodeURL)) {
  2422. const script = baseScript.code_url.includes('.js?version=') ? baseScript : (await getScriptData(scriptID));
  2423. scriptCodeURL = script.code_url;
  2424. }
  2425.  
  2426. if (scriptCodeURL && isLibraryURLWithVersion(scriptCodeURL)) {
  2427.  
  2428.  
  2429. const code_url = fixLibraryCodeURL(scriptCodeURL);
  2430.  
  2431. const button = addInstallButton(element, code_url);
  2432. button.textContent = `Copy URL`;
  2433. button.addEventListener('click', function (evt) {
  2434.  
  2435. const target = (evt || 0).target;
  2436. if (!target) return;
  2437.  
  2438. let a = target.nodeName === 'A' ? target : target.querySelector('a[href]');
  2439.  
  2440. if (!a) return;
  2441. let href = target.getAttribute('href');
  2442. if (!href) return;
  2443.  
  2444. evt.preventDefault();
  2445.  
  2446. copyText(href);
  2447.  
  2448.  
  2449. });
  2450.  
  2451. }
  2452.  
  2453.  
  2454. } else {
  2455.  
  2456.  
  2457. if (!baseScript || !baseScript.code_url || !baseScript.version) return;
  2458. const button = addInstallButton(element, baseScript.code_url);
  2459. button.classList.add('install-status-checking');
  2460. button.textContent = `${installLabel()} ${baseScript.version}`;
  2461. const script = baseScript && baseScript.name && baseScript.namespace ? baseScript : (await getScriptData(scriptID));
  2462. if (!script) return;
  2463.  
  2464. const installed = await isInstalled(script);
  2465. const version = (
  2466. baseScript.version && script.version && compareVersions(baseScript.version, script.version) === 1
  2467. ) ? baseScript.version : script.version;
  2468.  
  2469. const update = compareVersions(version, installed); // NaN 1 -1 0
  2470. const label = installLabel(update);
  2471. button.textContent = `${label} ${version}`;
  2472. button.classList.remove('install-status-checking');
  2473.  
  2474.  
  2475. }
  2476.  
  2477. }
  2478.  
  2479. const updateReqStoresWithElementsOrder = (x) => {
  2480. try {
  2481. const reqStoresA_ = reqStoresA;
  2482. const reqStoresB_ = reqStoresB;
  2483. const order2 = [...reqStoresA_.keys()];
  2484. const order3 = [...reqStoresB_.keys()];
  2485. const orders1 = x;
  2486. const orders = new Set([...orders1, ...order2, ...order3]);
  2487. const reqStoresA2 = new Map();
  2488. const reqStoresB2 = new Map();
  2489. for (const id of orders) {
  2490. const reqA = reqStoresA_.get(id);
  2491. if (reqA) reqStoresA2.set(id, reqA);
  2492. const reqB = reqStoresB_.get(id);
  2493. if (reqB) reqStoresB2.set(id, reqB);
  2494. }
  2495. reqStoresA = reqStoresA2;
  2496. reqStoresB = reqStoresB2;
  2497. reqStoresA_.clear();
  2498. reqStoresB_.clear();
  2499. } catch (e) {
  2500. console.warn(e)
  2501. }
  2502. };
  2503.  
  2504. let lastIdArrString = '';
  2505.  
  2506. const foundScriptList = async (scriptList) => {
  2507.  
  2508. // add options and style for blacklisted/hidden scripts
  2509. if (gmc.get('hideBlacklistedScripts') || gmc.get('hideHiddenScript')) {
  2510. addOptions(scriptList);
  2511. }
  2512.  
  2513. mutationRunner(() => {
  2514. if (!scriptList || scriptList.isConnected !== true) return;
  2515. const scriptElements = scriptList.querySelectorAll('li[data-script-id]:not([e8kk])');
  2516. for (const element of scriptElements) {
  2517. element.setAttribute('e8kk', '1');
  2518.  
  2519. const scriptID = +element.getAttribute('data-script-id');
  2520. if (!(scriptID > 0)) continue;
  2521.  
  2522. // blacklisted scripts
  2523. if (gmc.get('nonLatins')) hideBlacklistedScript(element, 'nonLatins');
  2524. if (gmc.get('blacklist')) hideBlacklistedScript(element, 'blacklist');
  2525. if (gmc.get('customBlacklist')) hideBlacklistedScript(element, 'customBlacklist');
  2526.  
  2527. // hidden scripts
  2528. if (gmc.get('hideHiddenScript')) hideHiddenScript(element, scriptID, true);
  2529.  
  2530. // install button
  2531. if (gmc.get('showInstallButton')) {
  2532. showInstallButton(scriptID, element)
  2533. }
  2534.  
  2535. }
  2536.  
  2537. const idArr = [...scriptList.querySelectorAll('li[data-script-id]')].map(e => +e.getAttribute('data-script-id'));
  2538. const idArrString = idArr.join(',');
  2539. if (lastIdArrString !== idArrString) {
  2540. lastIdArrString = idArrString;
  2541. updateReqStoresWithElementsOrder(idArr);
  2542. }
  2543.  
  2544. }, scriptList, { subtree: true, childList: true });
  2545.  
  2546. }
  2547.  
  2548. const foundDiscussionList = (discussionsList) => {
  2549. targetHiddenRecentDateTime = Date.now() - valHideRecentUsersWithin * 3600000;
  2550. mutationRunner(() => {
  2551. if (!discussionsList || discussionsList.isConnected !== true) return;
  2552. const scriptElements = discussionsList.querySelectorAll('.discussion-list-item:not([e8kk])');
  2553. for (const element of scriptElements) {
  2554. element.setAttribute('e8kk', '1');
  2555.  
  2556. // blacklisted scripts
  2557. if (gmc.get('hideHiddenScript')) hideBlacklistedDiscussion(element, 'hiddenList');
  2558.  
  2559. let t;
  2560. let userId = 0;
  2561. if (t = element.querySelector('a.user-link[href*="/users/"]')) {
  2562. const m = /\/users\/(\d+)/.exec(`${t.getAttribute('href')}`);
  2563. if (m) {
  2564. userId = +m[1];
  2565. }
  2566. }
  2567. if (userId > 0) {
  2568. determineRecentUserAsync(userId).then((isNewUser) => {
  2569. element.classList.toggle('discussion-item-by-recent-user', isNewUser);
  2570. });
  2571. }
  2572. let discussionId = 0;
  2573. if (t = element.querySelector('a.discussion-title[href*="/discussions/')) {
  2574. const m = /\/\w+\/(\d+)/.exec(`${t.getAttribute('href')}`);
  2575. if (m) {
  2576. discussionId = +m[1];
  2577. }
  2578. }
  2579. let btnContainer = null;
  2580. const meta = element.querySelector('div.discussion-meta');
  2581. if (meta) {
  2582. btnContainer = document.createElement('additional-buttons');
  2583. meta.appendChild(btnContainer);
  2584. }
  2585. if (btnContainer) {
  2586. if (discussionId > 0) {
  2587. const btn = document.createElement('a');
  2588. btn.classList = 'discussion-list-item-report-comment'
  2589. btn.textContent = 'Report Comment';
  2590. btnContainer.appendChild(btn);
  2591. const m = /^(https?:\/\/[a-z-]{10,15}\.org\/(([a-z]{2,3}(-[a-zA-Z0-9]{2,3})?)\/)?)\w+/.exec(location.href);
  2592. if (m) {
  2593. btn.href = `${m[1]}reports/new?item_class=discussion&item_id=${discussionId}`;
  2594. }
  2595. }
  2596. }
  2597. }
  2598. }, discussionsList, { subtree: true, childList: true });
  2599. }
  2600.  
  2601. const foundScriptDiscussionList = (discussionsList) => {
  2602. targetHiddenRecentDateTime = Date.now() - valHideRecentUsersWithin * 3600000;
  2603. mutationRunner(() => {
  2604. if (!discussionsList || discussionsList.isConnected !== true) return;
  2605. const scriptElements = discussionsList.querySelectorAll('.discussion-list-item:not([e8kk])');
  2606. for (const element of scriptElements) {
  2607. element.setAttribute('e8kk', '1');
  2608. let t;
  2609. let userId = 0;
  2610. if (t = element.querySelector('a.user-link[href*="/users/"]')) {
  2611. const m = /\/users\/(\d+)/.exec(`${t.getAttribute('href')}`);
  2612. if (m) {
  2613. userId = +m[1];
  2614. }
  2615. }
  2616. if (userId > 0) {
  2617. determineRecentUserAsync(userId).then((isNewUser) => {
  2618. element.classList.toggle('discussion-item-by-recent-user', isNewUser);
  2619. });
  2620. }
  2621. let discussionId = 0;
  2622. if (t = element.querySelector('a.discussion-title[href*="/discussions/')) {
  2623. const m = /\/\w+\/(\d+)/.exec(`${t.getAttribute('href')}`);
  2624. if (m) {
  2625. discussionId = +m[1];
  2626. }
  2627. }
  2628. let btnContainer = null;
  2629. const meta = element.querySelector('div.discussion-meta');
  2630. if(meta){
  2631. btnContainer = document.createElement('additional-buttons');
  2632. meta.appendChild(btnContainer);
  2633. }
  2634. if (btnContainer) {
  2635. if (discussionId > 0) {
  2636. const btn = document.createElement('a');
  2637. btn.classList = 'discussion-list-item-report-comment'
  2638. btn.textContent = 'Report Comment';
  2639. btnContainer.appendChild(btn);
  2640. const m = /^(https?:\/\/[a-z-]{10,15}\.org\/(([a-z]{2,3}(-[a-zA-Z0-9]{2,3})?)\/)?)\w+/.exec(location.href);
  2641. if (m) {
  2642. btn.href = `${m[1]}reports/new?item_class=discussion&item_id=${discussionId}`;
  2643. }
  2644. }
  2645. }
  2646. }
  2647. }, discussionsList, { subtree: true, childList: true });
  2648. }
  2649.  
  2650. let promiseScriptCheckResolve = null;
  2651. const promiseScriptCheck = new Promise(resolve => {
  2652. promiseScriptCheckResolve = resolve
  2653. });
  2654.  
  2655. const milestoneNotificationFn = async (o) => {
  2656.  
  2657. const { userLink, userID } = o;
  2658.  
  2659.  
  2660. const milestones = gmc.get('milestoneNotification').replace(/\s/g, '').split(',').map(Number);
  2661.  
  2662. if (!userID) return;
  2663.  
  2664. await new Promise(resolve => setTimeout(resolve, 800)); // delay for reducing server burden
  2665. await new Promise(resolve => requestAnimationFrame(resolve)); // foreground
  2666.  
  2667. const userData = await getUserData(+userID.match(/\d+(?=\D)/g));
  2668. if (!userData) return;
  2669.  
  2670. const [totalInstalls, lastMilestone] = await Promise.all([
  2671. getTotalInstalls(userData),
  2672. GM.getValue('lastMilestone', 0)]);
  2673.  
  2674. const milestone = milestones.filter(milestone => totalInstalls >= milestone).pop();
  2675.  
  2676. UU.log(`total installs are "${totalInstalls}", milestone reached is "${milestone}", last milestone reached is "${lastMilestone}"`);
  2677.  
  2678. if (milestone <= lastMilestone) return;
  2679.  
  2680. if (milestone && milestone >= 0) {
  2681.  
  2682.  
  2683. GM.setValue('lastMilestone', milestone);
  2684.  
  2685. const lang = document.documentElement.lang;
  2686. const text = (locales[lang] ? locales[lang].milestone : locales.en.milestone).replace('$1', milestone.toLocaleString());
  2687.  
  2688. if (typeof GM.notification === 'function') {
  2689. GM.notification({
  2690. text,
  2691. title: GM.info.script.name,
  2692. image: logo,
  2693. onclick: () => {
  2694. window.location = `https://${window.location.hostname}${userID}#user-script-list-section`;
  2695. }
  2696. });
  2697. } else {
  2698. UU.alert(text);
  2699. }
  2700.  
  2701. }
  2702.  
  2703. }
  2704. const onReady = async () => {
  2705.  
  2706. try {
  2707.  
  2708. const gminfo = GM.info || 0;
  2709. if (gminfo) {
  2710.  
  2711. const gminfoscript = gminfo.script || 0;
  2712.  
  2713.  
  2714. const scriptHandlerObject = {
  2715. scriptHandler: gminfo.scriptHandler || '',
  2716. scriptName: gminfoscript.name || '', // not name_i18n
  2717. scriptVersion: gminfoscript.version || '',
  2718. scriptNamespace: gminfoscript.namespace || '',
  2719. communicationId
  2720. };
  2721.  
  2722.  
  2723. wincomm.hook('_$GreasyFork$Msg$OnScriptInstallFeedback',
  2724. {
  2725.  
  2726. ready: (d, evt) => promiseScriptCheckResolve(d),
  2727. userScriptManagerNotDetected: (d, evt) => promiseScriptCheckResolve(null),
  2728. 'installedVersion.res': wincomm.handleResponse
  2729.  
  2730.  
  2731. })
  2732.  
  2733.  
  2734. document.head.appendChild(document.createElement('script')).textContent = `;(${mWindow.contentScriptText})(${JSON.stringify(scriptHandlerObject)}, ${WinComm.createInstance});`;
  2735.  
  2736.  
  2737. }
  2738.  
  2739. addSettingsToMenu();
  2740.  
  2741. setTimeout(() => {
  2742. getRafPromise().then(() => {
  2743. let installBtn = document.querySelector('a[data-script-id][data-script-version]')
  2744. let scriptID = installBtn && installBtn.textContent ? +installBtn.getAttribute('data-script-id') : 0;
  2745. if (scriptID > 0) {
  2746. getScriptData(scriptID, true);
  2747. } else {
  2748. const userLink = document.querySelector('#site-nav .user-profile-link a[href]');
  2749. let userID = userLink ? userLink.getAttribute('href') : '';
  2750. userID = userID ? /users\/(\d+)/.exec(userID) : null;
  2751. if (userID) userID = userID[1];
  2752. if (userID) {
  2753. userID = +userID;
  2754. if (userID > 0) {
  2755. getUserData(userID, true);
  2756. }
  2757. }
  2758. }
  2759. });
  2760. }, 740);
  2761.  
  2762. const userLink = document.querySelector('.user-profile-link a[href]');
  2763. const userID = userLink ? userLink.getAttribute('href') : undefined;
  2764.  
  2765. const urlMatch = (url1, url2) => {
  2766. url1 = `${url1}`
  2767. url2 = `${url2}`;
  2768. if (url1.includes(location.hostname)) {
  2769. url1 = url1.replace(`https://${location.hostname}/`, '/')
  2770. url1 = url1.replace(`http://${location.hostname}/`, '/')
  2771. url1 = url1.replace(/^\/+/, '/')
  2772. } else if (!url1.startsWith('/')) {
  2773. url1 = `/${url1}`;
  2774. }
  2775. if (url2.includes(location.hostname)) {
  2776. url2 = url2.replace(`https://${location.hostname}/`, '/')
  2777. url2 = url2.replace(`http://${location.hostname}/`, '/')
  2778. url2 = url2.replace(/^\/+/, '/')
  2779. } else if (!url2.startsWith('/')) {
  2780. url2 = `/${url2}`;
  2781. }
  2782. url1 = url1.replace(/\?\w+=\w+(&\w+=\w+)*$/, '');
  2783. url2 = url2.replace(/\?\w+=\w+(&\w+=\w+)*$/, '');
  2784. return url1.toLowerCase() === url2.toLowerCase();
  2785. }
  2786.  
  2787. UU.addStyle(mWindow.pageCSS);
  2788.  
  2789.  
  2790. const elementLookup = (selector, fn) => {
  2791. const elm0 = document.querySelector(selector);
  2792. if (elm0) {
  2793. fn(elm0);
  2794. } else {
  2795. const timeout = Date.now() + 3000;
  2796. (new MutationObserver((_, observer) => {
  2797. const elm = document.querySelector(selector);
  2798. if (elm && elm.childElementCount >= 1) {
  2799. observer.disconnect();
  2800. observer.takeRecords();
  2801. fn(elm);
  2802. } else if (Date.now() > timeout) {
  2803. observer.disconnect();
  2804. observer.takeRecords();
  2805. }
  2806. })).observe(document, { subtree: true, childList: true });
  2807. }
  2808. };
  2809.  
  2810. // blacklisted scripts / hidden scripts / install button
  2811.  
  2812. const isPageUnderScript = location.pathname.includes('/scripts/');
  2813. const pageType_ = /\/([a-z-]+)$/.exec(window.location.pathname);
  2814. const pageType = pageType_ ? pageType_[1] : '';
  2815. const isDiscussionListPage = !isPageUnderScript && (pageType === 'discussions' || (pageType_ && /\/discussions\/[a-z-]+$/.test(location.pathname)));
  2816. const isFeedbackListPage = isPageUnderScript && pageType === 'feedback';
  2817. const isScriptListPage = !isPageUnderScript && pageType === 'scripts';
  2818. const isUserIDPage = !isPageUnderScript && urlMatch(window.location.pathname, userID);
  2819. if (!isUserIDPage && !isDiscussionListPage && !isFeedbackListPage && (gmc.get('hideBlacklistedScripts') || gmc.get('hideHiddenScript') || gmc.get('showInstallButton'))) {
  2820.  
  2821. if (isScriptListPage) {
  2822. elementLookup('.script-list', foundScriptList);
  2823. } else if (isPageUnderScript) {
  2824.  
  2825. // hidden scripts on details page
  2826. const installLinkElement = document.querySelector('#script-info .install-link[data-script-id]');
  2827.  
  2828. if (installLinkElement) {
  2829. setupInstallLink(installLinkElement);
  2830. if (gmc.get('hideHiddenScript')) {
  2831. const id = +installLinkElement.getAttribute('data-script-id');
  2832. hideHiddenScript(document.querySelector('#script-info'), id, false);
  2833. }
  2834. installLinkElement.addEventListener('click', async function (e) {
  2835. if (e && e.isTrusted && location.pathname.includes('/scripts/')) {
  2836.  
  2837. await new Promise(r => setTimeout(r, 800));
  2838. await new Promise(r => window.requestAnimationFrame(r));
  2839. await new Promise(r => setTimeout(r, 100));
  2840. // let ethicalads497 = 'ethicalads' in window ? window.ethicalads : undefined;
  2841. // window.ethicalads = { wait: new Promise() }
  2842. document.dispatchEvent(new Event("DOMContentLoaded"));
  2843. document.documentElement.dispatchEvent(new Event("turbo:load"));
  2844. // if (ethicalads497 === undefined) delete window.ethicalads; else window.ethicalads = ethicalads497;
  2845. }
  2846. })
  2847. }
  2848.  
  2849.  
  2850. }
  2851.  
  2852. } else if (isDiscussionListPage) {
  2853. elementLookup('.discussion-list', foundDiscussionList);
  2854. } else if (isFeedbackListPage) {
  2855. elementLookup('.script-discussion-list', foundScriptDiscussionList);
  2856. }
  2857.  
  2858. // total installs
  2859. if (gmc.get('showTotalInstalls') && document.querySelector('#user-script-list')) {
  2860. const dailyInstalls = [];
  2861. const totalInstalls = [];
  2862.  
  2863. const dailyInstallElements = document.querySelectorAll('#user-script-list li dd.script-list-daily-installs');
  2864. for (const element of dailyInstallElements) {
  2865. dailyInstalls.push(parseInt(element.textContent.replace(/\D/g, ''), 10));
  2866. }
  2867.  
  2868. const totalInstallElements = document.querySelectorAll('#user-script-list li dd.script-list-total-installs');
  2869. for (const element of totalInstallElements) {
  2870. totalInstalls.push(parseInt(element.textContent.replace(/\D/g, ''), 10));
  2871. }
  2872.  
  2873. const dailyInstallsSum = dailyInstalls.reduce((a, b) => a + b, 0);
  2874. const totalInstallsSum = totalInstalls.reduce((a, b) => a + b, 0);
  2875.  
  2876. const convertLi = (li) => {
  2877. if (!li) return null;
  2878. const a = li.firstElementChild
  2879. if (a === null) return li;
  2880. if (a === li.lastElementChild && a.nodeName === 'A') return a;
  2881. return null;
  2882. }
  2883.  
  2884. const plusSign = document.querySelector('#user-script-list-section a[rel="next"][href*="page="], #user-script-list-section a[rel="prev"][href*="page="]') ? '+' : '';
  2885.  
  2886. const dailyOption = convertLi(document.querySelector('#script-list-sort .list-option:nth-child(1)'));
  2887. dailyOption && dailyOption.insertAdjacentHTML('beforeend', `<span> (${dailyInstallsSum.toLocaleString()}${plusSign})</span>`);
  2888.  
  2889. const totalOption = convertLi(document.querySelector('#script-list-sort .list-option:nth-child(2)'));
  2890. totalOption && totalOption.insertAdjacentHTML('beforeend', `<span> (${totalInstallsSum.toLocaleString()}${plusSign})</span>`);
  2891. }
  2892.  
  2893. // milestone notification
  2894. if (gmc.get('milestoneNotification')) {
  2895. milestoneNotificationFn({ userLink, userID });
  2896. }
  2897.  
  2898. if (isScriptFirstUse) GM.setValue('firstUse', false).then(() => {
  2899. gmc.open();
  2900. });
  2901.  
  2902. if (fixLibraryScriptCodeLink) {
  2903.  
  2904.  
  2905. let xpath = "//code[contains(text(), '.js?version=') or contains(text(), '// @require https://')]";
  2906. let snapshot = document.evaluate(xpath, document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
  2907.  
  2908. for (let i = 0; i < snapshot.snapshotLength; i++) {
  2909. let element = snapshot.snapshotItem(i);
  2910. if (element.firstElementChild) continue;
  2911. element.textContent = element.textContent.replace(/\bhttps:\/\/(cn-greasyfork|greasyfork|sleazyfork)\.org\/scripts\/\d+\-[^\/]+\/code\/[^\.]+\.js\?version=\d+\b/, (_) => {
  2912. return fixLibraryCodeURL(_);
  2913. });
  2914. element.parentNode.insertBefore(document.createTextNode('\u200B'), element);
  2915. element.style.display = 'inline-flex';
  2916. setClickToSelect(element);
  2917. }
  2918.  
  2919.  
  2920. }
  2921.  
  2922.  
  2923.  
  2924.  
  2925. if (addAdditionInfoLengthHint && location.pathname.includes('/scripts/') && location.pathname.includes('/versions')) {
  2926.  
  2927. function contentLength(text) {
  2928. return text.replace(/\n/g, ' ').length;
  2929. }
  2930. function contentLengthMax() {
  2931. return 50000;
  2932. }
  2933. let _spanContent = null;
  2934. function updateText(ainfo, span) {
  2935. const value = ainfo.value;
  2936. if (typeof value !== 'string') return;
  2937.  
  2938. if (_spanContent !== value) {
  2939. _spanContent = value;
  2940. span.textContent = `Text Length: ${contentLength(value)} / ${contentLengthMax()}`;
  2941.  
  2942.  
  2943. }
  2944. }
  2945. function onChange(evt) {
  2946. let ainfo = (evt || 0).target;
  2947. if (!ainfo) return;
  2948. let span = ainfo.parentNode.querySelector('.script-version-ainfo-span');
  2949. if (!span) return;
  2950.  
  2951. updateText(ainfo, span);
  2952.  
  2953. }
  2954. function kbEvent(evt) {
  2955. Promise.resolve().then(() => {
  2956. onChange(evt);
  2957.  
  2958. })
  2959. }
  2960. for (const ainfo of document.querySelectorAll('textarea[id^="script-version-additional-info"]')) {
  2961. let span = document.createElement('span');
  2962. span.classList.add('script-version-ainfo-span');
  2963. ainfo.addEventListener('change', onChange, false);
  2964. ainfo.addEventListener('keydown', kbEvent, false);
  2965. ainfo.addEventListener('keypress', kbEvent, false);
  2966. ainfo.addEventListener('keyup', kbEvent, false);
  2967. updateText(ainfo, span);
  2968. ainfo.parentNode.insertBefore(span, ainfo.nextSibling);
  2969.  
  2970.  
  2971. }
  2972.  
  2973.  
  2974. }
  2975.  
  2976. } catch (e) {
  2977. console.log(e);
  2978. }
  2979.  
  2980.  
  2981.  
  2982. }
  2983.  
  2984.  
  2985.  
  2986.  
  2987. Promise.resolve().then(() => {
  2988. if (document.readyState !== 'loading') {
  2989. onReady();
  2990. } else {
  2991. window.addEventListener("DOMContentLoaded", onReady, false);
  2992. }
  2993. });
  2994.  
  2995. })();

QingJ © 2025

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