- // ==UserScript==
- // @name Greasy Fork++
- // @author CY Fung <https://greasyfork.org/users/371179> & Davide <iFelix18@protonmail.com>
- // @namespace https://github.com/iFelix18
- // @icon https://www.google.com/s2/favicons?domain=https://greasyfork.org
- // @description Adds various features and improves the Greasy Fork experience
- // @description:de Fügt verschiedene Funktionen hinzu und verbessert das Greasy Fork-Erlebnis
- // @description:es Agrega varias funciones y mejora la experiencia de Greasy Fork
- // @description:fr Ajoute diverses fonctionnalités et améliore l'expérience Greasy Fork
- // @description:it Aggiunge varie funzionalità e migliora l'esperienza di Greasy Fork
- // @description:ru Добавляет различные функции и улучшает работу с Greasy Fork
- // @description:zh-CN 添加各种功能并改善 Greasy Fork 体验
- // @description:zh-TW 加入多種功能並改善Greasy Fork的體驗
- // @description:ja Greasy Forkの体験を向上させる様々な機能を追加
- // @description:ko Greasy Fork 경험을 향상시키고 다양한 기능을 추가
- // @copyright 2023, CY Fung (https://greasyfork.org/users/371179); 2021, Davide (https://github.com/iFelix18)
- // @license MIT
- // @version 3.0.10
- // @require https://fastly.jsdelivr.net/gh/sizzlemctwizzle/GM_config@06f2015c04db3aaab9717298394ca4f025802873/gm_config.js
- // @require https://fastly.jsdelivr.net/npm/@violentmonkey/shortcut@1.3.0/dist/index.min.js
- // @match *://greasyfork.org/*
- // @match *://sleazyfork.org/*
- // @connect greasyfork.org
- // @compatible chrome
- // @compatible edge
- // @compatible firefox
- // @compatible safari
- // @compatible brave
- // @grant GM.deleteValue
- // @grant GM.getValue
- // @grant GM.notification
- // @grant GM.registerMenuCommand
- // @grant GM.setValue
- // @run-at document-start
- // @inject-into page
- // ==/UserScript==
-
- /* global GM_config, VM, GM */
-
- // -------- UU Fucntion - original code: https://fastly.jsdelivr.net/npm/@ifelix18/utils@6.5.0/lib/index.min.js --------
- // optimized by CY Fung to remove $ dependency and observe creation
- const UU = (function () {
- const scriptName = GM.info.script.name;
- const scriptVersion = GM.info.script.version;
- const authorMatch = /^(.*?)\s<\S[^\s@]*@\S[^\s.]*\.\S+>$/.exec(GM.info.script.author);
- const author = authorMatch ? authorMatch[1] : GM.info.script.author;
- let scriptId = scriptName.toLowerCase().replace(/\s/g, "-");
- let loggingEnabled = false;
-
- const log = (message) => {
- if (loggingEnabled) {
- console.log(`${scriptName}:`, message);
- }
- };
-
- const error = (message) => {
- console.error(`${scriptName}:`, message);
- };
-
- const warn = (message) => {
- console.warn(`${scriptName}:`, message);
- };
-
- const alert = (message) => {
- window.alert(`${scriptName}: ${message}`);
- };
-
- /** @param {string} text */
- const short = (text, length) => {
- const s = text.split(" ");
- const l = Number(length);
- return s.length > l
- ? `${s.slice(0, l).join(" ")} [...]`
- : text;
- };
-
- const addStyle = (css) => {
- const head = document.head || document.querySelector("head");
- const style = document.createElement("style");
- style.textContent = css;
- head.appendChild(style);
- };
-
- const init = async (options = {}) => {
- scriptId = options.id || scriptId;
- loggingEnabled = typeof options.logging === "boolean" ? options.logging : false;
- console.info(
- `%c${scriptName}\n%cv${scriptVersion}${author ? ` by ${author}` : ""} is running!`,
- "color:red;font-weight:700;font-size:18px;text-transform:uppercase",
- ""
- );
- };
-
- return {
- init,
- log,
- error,
- warn,
- alert,
- short,
- addStyle
- };
- })();
-
- // -------- UU Fucntion - original code: https://fastly.jsdelivr.net/npm/@ifelix18/utils@6.5.0/lib/index.min.js --------
-
-
- const mWindow = (() => {
-
-
- const fields = {
- hideBlacklistedScripts: {
- 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>',
- section: ['Features'],
- labelPos: 'right',
- type: 'checkbox',
- default: true
- },
- hideHiddenScript: {
- 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',
- labelPos: 'right',
- type: 'checkbox',
- default: true
- },
- showInstallButton: {
- label: 'Install button:<br><span>Add to the scripts list a button to install the script directly</span>',
- labelPos: 'right',
- type: 'checkbox',
- default: true
- },
- showTotalInstalls: {
- label: 'Installations:<br><span>Shows the number of daily and total installations on the user profile</span>',
- labelPos: 'right',
- type: 'checkbox',
- default: true
- },
- milestoneNotification: {
- 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>',
- labelPos: 'left',
- type: 'text',
- title: 'Separate milestones with a comma!',
- size: 150,
- default: '10, 100, 500, 1000, 2500, 5000, 10000, 100000, 1000000'
- },
- nonLatins: {
- label: 'Non-Latin:<br><span>This list blocks all scripts with non-Latin characters in the title/description</span>',
- section: ['Lists'],
- labelPos: 'right',
- type: 'checkbox',
- default: false // not true
- },
- blacklist: {
- label: 'Blacklist:<br><span>A "non-opinionable" list that blocks all scripts with emoji in the title/description, references to "bots", "cheats" and some online game sites, and other "bullshit"</span>',
- labelPos: 'right',
- type: 'checkbox',
- default: true
- },
- customBlacklist: {
- 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>',
- labelPos: 'left',
- type: 'text',
- title: 'Separate unwanted words with a comma!',
- size: 150,
- default: ''
- },
- hiddenList: {
- label: 'Hidden Scripts:<br><span>Block individual undesired scripts by their unique IDs<br>Separate IDs with a comma</span>',
- labelPos: 'left',
- type: 'textarea',
- title: 'Separate IDs with a comma!',
- default: '',
- save: false
- },
- logging: {
- label: 'Logging',
- section: ['Developer options'],
- labelPos: 'right',
- type: 'checkbox',
- default: false
- },
- debugging: {
- label: 'Debugging',
- labelPos: 'right',
- type: 'checkbox',
- default: false
- }
- }
-
- const logo = ''
-
- const locales = { /* cSpell: disable */
- de: {
- downgrade: 'Auf zurückstufen',
- hide: '❌ Dieses skript ausblenden',
- install: 'Installieren',
- notHide: '✔️ Dieses skript nicht ausblenden',
- milestone: 'Herzlichen Glückwunsch, Ihre Skripte haben den Meilenstein von insgesamt $1 Installationen überschritten!',
- reinstall: 'Erneut installieren',
- update: 'Auf aktualisieren'
- },
- en: {
- downgrade: 'Downgrade to',
- hide: '❌ Hide this script',
- install: 'Install',
- notHide: '✔️ Not hide this script',
- milestone: 'Congrats, your scripts got over the milestone of $1 total installs!',
- reinstall: 'Reinstall',
- update: 'Update to'
- },
- es: {
- downgrade: 'Degradar a',
- hide: '❌ Ocultar este script',
- install: 'Instalar',
- notHide: '✔️ No ocultar este script',
- milestone: '¡Felicidades, sus scripts superaron el hito de $1 instalaciones totales!',
- reinstall: 'Reinstalar',
- update: 'Actualizar a'
- },
- fr: {
- downgrade: 'Revenir à',
- hide: '❌ Cacher ce script',
- install: 'Installer',
- notHide: '✔️ Ne pas cacher ce script',
- milestone: 'Félicitations, vos scripts ont franchi le cap des $1 installations au total!',
- reinstall: 'Réinstaller',
- update: 'Mettre à'
- },
- it: {
- downgrade: 'Riporta a',
- hide: '❌ Nascondi questo script',
- install: 'Installa',
- notHide: '✔️ Non nascondere questo script',
- milestone: 'Congratulazioni, i tuoi script hanno superato il traguardo di $1 installazioni totali!',
- reinstall: 'Reinstalla',
- update: 'Aggiorna a'
- },
- ru: {
- downgrade: 'Откатить до',
- hide: '❌ Скрыть этот скрипт',
- install: 'Установить',
- notHide: '✔️ Не скрывать этот сценарий',
- milestone: 'Поздравляем, ваши скрипты преодолели рубеж в $1 установок!',
- reinstall: 'Переустановить',
- update: 'Обновить до'
- },
- 'zh-CN': {
- downgrade: '降级到',
- hide: '❌ 隐藏此脚本',
- install: '安装',
- notHide: '✔️ 不隐藏此脚本',
- milestone: '恭喜,您的脚本超过了 $1 次总安装的里程碑!',
- reinstall: '重新安装',
- update: '更新到'
- },
- 'zh-TW': {
- downgrade: '降級至',
- hide: '❌ 隱藏此腳本',
- install: '安裝',
- notHide: '✔️ 不隱藏此腳本',
- milestone: '恭喜,您的腳本安裝總數已超過 $1!',
- reinstall: '重新安裝',
- update: '更新至'
- },
- 'ja': {
- downgrade: 'ダウングレードする',
- hide: '❌ このスクリプトを隠す',
- install: 'インストール',
- notHide: '✔️ このスクリプトを隠さない',
- milestone: 'おめでとうございます、あなたのスクリプトの合計インストール回数が $1 を超えました!',
- reinstall: '再インストール',
- update: '更新する'
- },
- 'ko': {
- downgrade: '다운그레이드하기',
- hide: '❌ 이 스크립트 숨기기',
- install: '설치',
- notHide: '✔️ 이 스크립트 숨기지 않기',
- milestone: '축하합니다, 스크립트의 총 설치 횟수가 $1을 넘었습니다!',
- reinstall: '재설치',
- update: '업데이트하기'
- }
-
- };
-
- const blacklist = [ /* cSpell: disable-next-line */
- '\\bagar((.)?io)?\\b', '\\bagma((.)?io)?\\b', '\\baimbot\\b', '\\barras((.)?io)?\\b', '\\bbot(s)?\\b', '\\bbubble((.)?am)?\\b', '\\bcheat(s)?\\b', '\\bdiep((.)?io)?\\b', '\\bfreebitco((.)?in)?\\b', '\\bgota((.)?io)?\\b', '\\bhack(s)?\\b', '\\bkrunker((.)?io)?\\b', '\\blostworld((.)?io)?\\b', '\\bmoomoo((.)?io)?\\b', '\\broblox(.com)?\\b', '\\bshell\\sshockers\\b', '\\bshellshock((.)?io)?\\b', '\\bshellshockers\\b', '\\bskribbl((.)?io)?\\b', '\\bslither((.)?io)?\\b', '\\bsurviv((.)?io)?\\b', '\\btaming((.)?io)?\\b', '\\bvenge((.)?io)?\\b', '\\bvertix((.)?io)?\\b', '\\bzombs((.)?io)?\\b', '\\p{Extended_Pictographic}'
- ];
-
-
- const settingsCSS = `
-
- #greasyfork-plus{
- --config-var-display: flex;
- }
- #greasyfork-plus *{
- font-family:Open Sans,sans-serif,Segoe UI Emoji !important;
- font-size:12px
- }
- #greasyfork-plus .section_header{
- background-color:#670000;
- background-image:linear-gradient(#670000,#900);
- border:1px solid transparent;
- color:#fff
- }
- #greasyfork-plus .field_label[class]{
- margin-bottom:4px
- }
- #greasyfork-plus .field_label[class] span{
- font-size:95%;
- font-style:italic;
- opacity:.8;
- }
- #greasyfork-plus .field_label[class] b{
- color:#670000
- }
- #greasyfork-plus_logging_var[class],
- #greasyfork-plus_debugging_var[class] {
- --config-var-display: inline-flex;
- }
- #greasyfork-plus #greasyfork-plus_logging_var label.field_label[class],
- #greasyfork-plus #greasyfork-plus_debugging_var label.field_label[class] {
- margin-bottom:0;
- align-self: center;
- }
- #greasyfork-plus .config_var[class]{
- display:var(--config-var-display);
- }
- #greasyfork-plus_customBlacklist_var[class],
- #greasyfork-plus_hiddenList_var[class],
- #greasyfork-plus_milestoneNotification_var[class]{
- flex-direction:column;
- margin-left:21px;
- }
-
- #greasyfork-plus_customBlacklist_var[class]::before,
- #greasyfork-plus_hiddenList_var[class]::before,
- #greasyfork-plus_milestoneNotification_var[class]::before{
- /* content: "◉"; */
- content: "◎";
- position: absolute;
- left: auto;
- top: auto;
- margin-left: -16px;
- }
- #greasyfork-plus_field_customBlacklist[class],
- #greasyfork-plus_field_milestoneNotification[class]{
- flex:1;
- }
- #greasyfork-plus_field_hiddenList[class]{
- box-sizing:border-box;
- overflow:hidden;
- resize:none;
- width:100%
- }
-
- #greasyfork-plus_wrapper {
- box-sizing: border-box;
- overflow: auto;
- max-height: calc(100vh - 72px);
- padding: 12px;
- /* overflow: auto; */
- scrollbar-gutter: both-edges;
- background: rgba(127,127,127,0.05);
- border: 1px solid rgba(127,127,127,0.5);
- }
-
- #greasyfork-plus_buttons_holder {
- position: fixed;
- bottom: 0;
- right: 0;
- margin: 0 12px 6px 0;
- }
-
- #greasyfork-plus .saveclose_buttons[class] {
- padding: 4px 14px;
- margin: 6px;
- }
-
- `;
-
- const pageCSS = `
-
- .script-list li.blacklisted{
- display:none;
- background:#321919;
- color:#e8e6e3
- }
- .script-list li.hidden{
- display:none;
- background:#321932;
- color:#e8e6e3
- }
- .script-list li.blacklisted a:not(.install-link),.script-list li.hidden a:not(.install-link){
- color:#ff8484
- }
- #script-info.hidden,#script-info.hidden .user-content{
- background:#321932;
- color:#e8e6e3
- }
- #script-info.hidden a:not(.install-link):not(.install-help-link){
- color:#ff8484
- }
- #script-info.hidden code{
- background-color:transparent
- }
- html {
- --block-btn-color:#111;
- --block-btn-bgcolor:#eee;
- }
- #script-info.hidden, #script-info.hidden .user-content {
- --block-btn-color:#eee;
- --block-btn-bgcolor:#111;
- }
-
- [style-54998]{
- float:right;
- transform: scale(0.7);
- text-decoration:none
- }
-
- [style-16377]{
- cursor:pointer;
- font-size:70%;
- white-space:nowrap;
- border: 1px solid #888;
- background: var(--block-btn-bgcolor, #eee);
- color: var(--block-btn-color);
- border-radius: 4px;
- padding: 0px 6px;
- }
- [style-77329] {
- cursor: pointer;
- margin-left: 1ex;
- white-space: nowrap;
- float: right;
- border: 1px solid #888;
- background: var(--block-btn-bgcolor, #eee);
- color: var(--block-btn-color);
- border-radius: 4px;
- padding: 0px 6px;
- }
-
- a#hyperlink-35389,
- a#hyperlink-40361,
- a#hyperlink-35389:visited,
- a#hyperlink-40361:visited,
- a#hyperlink-35389:hover,
- a#hyperlink-40361:hover,
- a#hyperlink-35389:focus,
- a#hyperlink-40361:focus,
- a#hyperlink-35389:active,
- a#hyperlink-40361:active {
-
- border: none !important;
- outline: none !important;
- box-shadow: none !important;
- appearance: none !important;
- background: none !important;
- color:inherit !important;
- }
-
- a#hyperlink-35389{
- opacity: var(--hyperlink-blacklisted-option-opacity);
-
- }
- a#hyperlink-40361{
- opacity: var(--hyperlink-hidden-option-opacity);
- }
-
-
- html {
-
- --hyperlink-blacklisted-option-opacity: 0.5;
- --hyperlink-hidden-option-opacity: 0.5;
- }
-
-
- .list-option.list-current[class] > a[href] {
-
- text-decoration:none;
- }
-
- html {
- --blacklisted-display: none;
- --hidden-display: none;
- }
-
- [blacklisted-shown] {
- --blacklisted-display: list-item;
- --hyperlink-blacklisted-option-opacity: 1;
- }
- [hidden-shown] {
- --hidden-display: list-item;
- --hyperlink-hidden-option-opacity: 1;
- }
-
- .script-list li.blacklisted{
- display: var(--blacklisted-display);
-
- }
-
- .script-list li.hidden{
- display: var(--hidden-display);
-
- }
-
- `
-
-
-
-
- return { fields, logo, locales, blacklist, settingsCSS, pageCSS }
-
-
-
- })();
-
- (async () => {
-
- function fixValue(key, def, test) {
- return GM.getValue(key, def).then((v) => test(v) || GM.deleteValue(key))
- }
-
- await Promise.all([
- fixValue('hiddenList', [], v => v && typeof v === 'object' && typeof v.length === 'number' && (v.length === 0 || typeof v[0] === 'number')),
- fixValue('lastMilestone', 0, v => v && typeof v === 'number' && v >= 0)
- ])
-
- const id = 'greasyfork-plus';
- const title = `${GM.info.script.name} v${GM.info.script.version} Settings`;
- const fields = mWindow.fields;
- const logo = mWindow.logo;
- const nonLatins = /[^\p{Script=Latin}\p{Script=Common}\p{Script=Inherited}]/gu;
- const blacklist = new RegExp(mWindow.blacklist.join('|'), 'giu');
- const hiddenList = await GM.getValue('hiddenList', []);
- const lang = document.documentElement.lang;
- const locales = mWindow.locales;
-
- const gmc = new GM_config({
- id,
- title,
- fields,
- css: mWindow.settingsCSS,
- events: {
- init: () => {
- gmc.initializedResolve && gmc.initializedResolve();
- gmc.initializedResolve = null;
- if (!Array.isArray(hiddenList)) {
- GM.deleteValue('hiddenList');
- setTimeout(() => window.location.reload(false), 500);
- }
-
- if (GM.info.scriptHandler !== 'Userscripts') {
- GM.registerMenuCommand('Configure', () => gmc.open());
- }
- },
- open: async (document) => {
- const textarea = document.querySelector(`#${id}_field_hiddenList`);
-
- const hiddenList = await GM.getValue('hiddenList', []);
- const unsavedHiddenList = gmc.get('hiddenList') !== '' ? gmc.get('hiddenList').split(',').map(Number) : [];
-
- if ((hiddenList.filter(item => !unsavedHiddenList.includes(item)).length > 0 || unsavedHiddenList.filter(item => !hiddenList.includes(item)).length > 0) && hiddenList.length !== 0) {
- gmc.fields.hiddenList.value = hiddenList.sort((a, b) => a - b).join(', ');
-
- gmc.close();
- gmc.open();
- }
-
- const resize = (target) => {
- target.style.height = '';
- target.style.height = `${target.scrollHeight}px`;
- };
-
- if (textarea) {
- resize(textarea);
- textarea.addEventListener('input', (event) => resize(event.target));
-
- }
- },
- save: async (forgotten) => {
- const unsavedHiddenList = forgotten.hiddenList !== '' ? forgotten.hiddenList.split(',').map(Number).filter((element) => element !== 0) : undefined;
-
- if (gmc.isOpen) {
- await GM.setValue('hiddenList', Array.from(unsavedHiddenList));
-
- UU.alert('settings saved');
- gmc.close();
- setTimeout(() => window.location.reload(false), 500);
- }
- }
- }
- });
- gmc.initialized = new Promise(r => (gmc.initializedResolve = r));
- await gmc.initialized.then();
-
- UU.init({ id, logging: gmc.get('logging') });
- UU.log(nonLatins);
- UU.log(blacklist);
- UU.log(hiddenList);
-
- const { register } = VM.shortcut;
- register('ctrl-alt-s', () => {
- gmc.open();
- });
- register('ctrl-alt-b', () => {
- toggleListDisplayingItem('blacklisted')
- // blacklistedToggled = !blacklistedToggled;
- // toggleElementVisibility('.script-list li.blacklisted');
- });
- register('ctrl-alt-h', () => {
- toggleListDisplayingItem('hidden')
- // hiddenToggled = !hiddenToggled;
- // toggleElementVisibility('.script-list li.hidden');
- });
-
- const addSettingsToMenu = () => {
- const menu = document.createElement('li');
- menu.classList.add(id);
- const link = document.createElement('a');
- link.setAttribute('href', '#');
- link.textContent = GM.info.script.name;
- menu.appendChild(link);
- let nav = document.querySelector('#site-nav > nav')
- nav && nav.insertBefore(menu, document.querySelector('#site-nav > nav > li:first-child'));
-
- menu.addEventListener('click', (e) => {
- e.preventDefault();
- e.stopPropagation();
- e.stopImmediatePropagation();
- gmc.open();
- });
- };
-
-
- const toggleListDisplayingItem = (t) => {
-
- const m = document.documentElement;
-
- const p = t + '-shown';
- let currentIsShown = m.hasAttribute(p)
- if (!currentIsShown) {
- m.setAttribute(p, '')
- } else {
- m.removeAttribute(p)
- }
-
- }
-
- const createListOptionGroup = () => {
-
- const html = `<div class="list-option-group" id="${id}-options">${GM.info.script.name} Lists:<ul>
- <li class="list-option blacklisted"><a href="#" id="hyperlink-35389"></a></li>
- <li class="list-option hidden"><a href="#" id="hyperlink-40361"></a></li>
- </ul></div>`;
- const firstOptionGroup = document.querySelector('.list-option-groups > div');
- firstOptionGroup && firstOptionGroup.insertAdjacentHTML('beforebegin', html);
-
- const blacklistedOption = document.querySelector(`#${id}-options li.blacklisted`);
- blacklistedOption && blacklistedOption.addEventListener('click', (evt) => {
- evt.preventDefault();
- toggleListDisplayingItem('blacklisted');
- }, false);
-
- const hiddenOption = document.querySelector(`#${id}-options li.hidden`);
- hiddenOption && hiddenOption.addEventListener('click', (evt) => {
- evt.preventDefault();
- toggleListDisplayingItem('hidden');
- }, false);
-
- }
-
- const addOptions = () => {
-
- const gn = () => {
-
- let aBlackList = document.querySelector('#hyperlink-35389');
- let aHidden = document.querySelector('#hyperlink-40361');
- if (!aBlackList || !aHidden) return;
- aBlackList.textContent = `Blacklisted scripts (${document.querySelectorAll('.script-list li.blacklisted').length})`;
- aHidden.textContent = `Hidden scripts (${document.querySelectorAll('.script-list li.hidden').length})`
-
- }
- const callback = (entries) => {
- if (entries && entries.length >= 1) requestAnimationFrame(gn);
- }
-
- const setupScriptList = async () => {
- let scriptList;
- let i = 8;
- while (i-- > 0) {
- scriptList = document.querySelector('.script-list li')
- if (scriptList) scriptList = scriptList.closest('.script-list')
- if (scriptList) break;
- await new Promise(r => requestAnimationFrame(r))
- }
- if (!scriptList) return;
- createListOptionGroup();
- const mo = new MutationObserver(callback);
- mo.observe(scriptList, { childList: true, subtree: true });
- gn();
- }
- setupScriptList();
-
- };
-
-
- /**
- * Get script data from Greasy Fork API
- *
- * @param {number} id Script ID
- * @returns {Promise} Script data
- */
- let networkMP1 = Promise.resolve();
- let networkMP2 = Promise.resolve();
- let previousIsCache = false;
- // let ss = [];
- // var sum = function(nums) {
- // var total = 0;
- // for (var i = 0, len = nums.length; i < len; i++) total += nums[i];
- // return total;
- // };
- const getScriptData = async (id, noCache) => {
- if (!(id >= 0)) return Promise.resolve()
- const url = `https://${window.location.hostname}/scripts/${id}.json`;
- return new Promise((resolve, reject) => {
-
- networkMP1 = networkMP1.then(() => new Promise(unlock => {
-
- const maxAgeInSeconds = 900;
- const rd = previousIsCache ? 1 : Math.floor(Math.random() * 80 + 80);
- let fetchStart = 0;
- new Promise(r => setTimeout(r, rd))
- .then(() => {
- fetchStart = Date.now();
- })
- .then(() => fetch(url, noCache ? {
- method: 'GET',
- cache: 'reload',
- credentials: 'omit',
- headers: new Headers({
- 'Cache-Control': `max-age=${maxAgeInSeconds}`,
- })
- } : {
- method: 'GET',
- cache: 'force-cache',
- credentials: 'omit',
- headers: new Headers({
- 'Cache-Control': `max-age=${maxAgeInSeconds}`,
- }),
- }))
- .then((response) => {
-
- let fetchStop = Date.now();
- // const dd = fetchStop - fetchStart;
- // dd (cache) = {min: 1, max: 8, avg: 3.7}
- // dd (normal) = {min: 136, max: 316, avg: 162.62}
-
- // ss.push(dd)
- // ss.maxValue = Math.max(...ss);
- // ss.minValue = Math.min(...ss);
- // ss.avgValue = sum(ss)/ss.length;
- // console.log(dd)
- // console.log(ss)
- previousIsCache = (fetchStop - fetchStart) < (3.7 + 162.62) / 2;
- UU.log(`${response.status}: ${response.url}`)
- // UU.log(response)
- if (response.ok === true) {
- unlock();
- return response.json()
- }
- if (response.status === 503) {
- return new Promise(r => setTimeout(r, 270 + rd)).then(() => {
- unlock();
- return getScriptData(id, true);
- });
- }
- console.warn(response);
- new Promise(r => setTimeout(r, 470)).then(unlock); // reload later
- })
- .then((data) => resolve(data))
- .catch((e) => {
- unlock();
- UU.log(id, url)
- console.warn(e)
- // reject(e)
- })
-
- })).catch(() => { })
-
- });
- }
-
- /**
- * Get user data from Greasy Fork API
- *
- * @param {string} userID User ID
- * @returns {Promise} User data
- */
- const getUserData = (userID, noCache) => {
-
- if (!(userID >= 0)) return Promise.resolve()
-
- const url = `https://${window.location.hostname}/users/${userID}.json`;
- return new Promise((resolve, reject) => {
-
-
- networkMP2 = networkMP2.then(() => new Promise(unlock => {
-
- const maxAgeInSeconds = 900;
- const rd = Math.floor(Math.random() * 80 + 80);
-
- new Promise(r => setTimeout(r, rd))
- .then(() => fetch(url, noCache ? {
- method: 'GET',
- cache: 'reload',
- credentials: 'omit',
- headers: new Headers({
- 'Cache-Control': `max-age=${maxAgeInSeconds}`,
- })
- } : {
- method: 'GET',
- cache: 'force-cache',
- credentials: 'omit',
- headers: new Headers({
- 'Cache-Control': `max-age=${maxAgeInSeconds}`,
- }),
- }))
- .then((response) => {
- UU.log(`${response.status}: ${response.url}`)
- if (response.ok === true) {
- unlock();
- return response.json()
- }
- if (response.status === 503) {
- return new Promise(r => setTimeout(r, 270 + rd)).then(() => {
- unlock();
- return getUserData(userID, true); // reload later
- });
- }
- console.warn(response);
- new Promise(r => setTimeout(r, 470)).then(unlock);
- })
- .then((data) => resolve(data))
- .catch((e) => {
- setTimeout(() => {
- unlock()
- }, 270)
- UU.log(userID, url)
- console.warn(e)
- // reject(e)
- })
-
-
-
- })).catch(() => { })
-
- });
- }
- const getTotalInstalls = (data) => {
- if (!data || !data.scripts) return;
- return new Promise((resolve, reject) => {
- const totalInstalls = [];
-
- data.scripts.forEach((element) => {
- totalInstalls.push(parseInt(element.total_installs, 10));
- });
-
- resolve(totalInstalls.reduce((a, b) => a + b, 0));
- });
- };
-
-
- const isInstalled = (name, namespace) => {
- return new Promise((resolve, reject) => {
- if (window.external && window.external.Violentmonkey) {
- window.external.Violentmonkey.isInstalled(name, namespace).then((data) => resolve(data));
- return;
- }
-
- if (window.external && window.external.Tampermonkey) {
- window.external.Tampermonkey.isInstalled(name, namespace, (data) => {
- (data.installed) ? resolve(data.version) : resolve();
- });
- return;
- }
-
- resolve();
- });
- };
-
- const compareVersions = (v1, v2) => {
- if (!v1 || !v2) return;
- if (v1 === null || v2 === null) return;
- if (v1 === v2) return 0;
-
- const sv1 = v1.split('.').map((index) => +index);
- const sv2 = v2.split('.').map((index) => +index);
-
- for (let index = 0; index < Math.max(sv1.length, sv2.length); index++) {
- if (sv1[index] > sv2[index]) return 1;
- if (sv1[index] < sv2[index]) return -1;
- }
-
- return 0;
- };
-
-
- /**
- * Return label for the hide script button
- *
- * @param {boolean} hidden Is hidden
- * @returns {string} Label
- */
- const blockLabel = (hidden) => {
- return hidden ? (locales[lang] ? locales[lang].notHide : locales.en.notHide) : (locales[lang] ? locales[lang].hide : locales.en.hide)
- }
-
- /**
- * Return label for the install button
- *
- * @param {number} update Update value
- * @returns {string} Label
- */
- const installLabel = (update) => {
- switch (update) {
- case undefined: {
- return locales[lang] ? locales[lang].install : locales.en.install
- }
- case 1: {
- return locales[lang] ? locales[lang].update : locales.en.update
- }
- case -1: {
- return locales[lang] ? locales[lang].downgrade : locales.en.downgrade
- }
- default: {
- return locales[lang] ? locales[lang].reinstall : locales.en.reinstall
- }
- }
- }
-
- const hideBlacklistedScript = (element, list) => {
- if (!element) return;
- const scriptLink = element.querySelector('.script-link')
-
- const name = scriptLink ? scriptLink.textContent : '';
- const descriptionElem = element.querySelector('.script-description')
- const description = descriptionElem ? descriptionElem.textContent : '';
-
- if (!name) return;
-
- switch (list) {
- case 'nonLatins':
- if ((nonLatins.test(name) || nonLatins.test(description)) && !element.classList.contains('blacklisted')) {
- element.classList.add('blacklisted', 'non-latins');
- if (gmc.get('hideBlacklistedScripts') && gmc.get('debugging')) {
- let scriptLink = element.querySelector('.script-link');
- if (scriptLink) { scriptLink.textContent += ' (non-latin)'; }
- }
- }
- break;
- case 'blacklist':
- if ((blacklist.test(name) || blacklist.test(description)) && !element.classList.contains('blacklisted')) {
- element.classList.add('blacklisted', 'blacklist');
- if (gmc.get('hideBlacklistedScripts') && gmc.get('debugging')) {
- let scriptLink = element.querySelector('.script-link');
- if (scriptLink) { scriptLink.textContent += ' (blacklist)'; }
- }
- }
- break;
- case 'customBlacklist': {
- const customBlacklist = new RegExp(gmc.get('customBlacklist').replace(/\s/g, '').split(',').join('|'), 'giu');
- if ((customBlacklist.test(name) || customBlacklist.test(description)) && !element.classList.contains('blacklisted')) {
- element.classList.add('blacklisted', 'custom-blacklist');
- if (gmc.get('hideBlacklistedScripts') && gmc.get('debugging')) {
- let scriptLink = element.querySelector('.script-link');
- if (scriptLink) { scriptLink.textContent += ' (custom-blacklist)'; }
- }
- }
- break;
- }
- default:
- UU.log('No blacklists');
- break;
- }
- };
-
- const hideHiddenScript = (element, id, list) => {
- id = +id;
- if (!(id >= 0)) return;
-
- const isInHiddenList = () => hiddenList.indexOf(id) !== -1;
- const updateScriptLink = (shouldHide) => {
- if (gmc.get('hideHiddenScript') && gmc.get('debugging')) {
- let scriptLink = element.querySelector('.script-link');
- if (scriptLink) {
- if (shouldHide) {
- scriptLink.innerHTML += ' (hidden)';
- } else {
- scriptLink.innerHTML = scriptLink.innerHTML.replace(' (hidden)', '');
- }
- }
- }
- };
-
- // Check for initial state and set it
- if (isInHiddenList()) {
- element.classList.add('hidden');
- updateScriptLink(true);
- }
-
- // Add button to hide the script
- const insertButtonHTML = (selector, html) => {
- const target = element.querySelector(selector);
- if (!target) return;
- let p = document.createElement('template');
- p.innerHTML = html;
- target.parentNode.insertBefore(p.content.firstChild, target.nextSibling);
- };
-
- const isHidden = element.classList.contains('hidden');
- const blockButtonHTML = `<span class=block-button role=button style-16377>${blockLabel(isHidden)}</span>`;
- const blockButtonHeaderHTML = `<span class=block-button role=button style-77329 style="">${blockLabel(isHidden)}</span>`;
-
- insertButtonHTML('.badge-js, .badge-css', blockButtonHTML);
- insertButtonHTML('header h2', blockButtonHeaderHTML);
-
- // Add event listener
- const button = element.querySelector('.block-button');
- if (button) {
- button.addEventListener('click', (event) => {
- event.stopPropagation();
- event.stopImmediatePropagation();
-
- if (!isInHiddenList()) {
- hiddenList.push(id);
- GM.setValue('hiddenList', hiddenList);
-
- element.classList.add('hidden');
- updateScriptLink(true);
-
- } else {
- const index = hiddenList.indexOf(id);
- hiddenList.splice(index, 1);
- GM.setValue('hiddenList', hiddenList);
-
- element.classList.remove('hidden');
- updateScriptLink(false);
- }
-
- const blockBtn = element.querySelector('.block-button');
- if (blockBtn) blockBtn.textContent = blockLabel(element.classList.contains('hidden'));
- });
- }
- };
-
- const insertButtonHTML = (element, selector, html) => {
- const target = element.querySelector(selector);
- if (!target) return;
- let p = document.createElement('template');
- p.innerHTML = html;
- target.parentNode.insertBefore(p.content.firstChild, target.nextSibling);
- };
-
- const addInstallButton = (element, url, label, version) => {
- insertButtonHTML(element, '.badge-js, .badge-css', `<a class="install-link" href="${url}" style-54998>${label} ${version}</a>`);
- };
-
- const showInstallButton = async (scriptID, element) => {
-
- const script = await getScriptData(scriptID);
- if (!script) return;
-
- const installed = await isInstalled(script.name, script.namespace)
-
- const update = compareVersions(script.version, installed);
- const label = installLabel(update);
- addInstallButton(element, script.code_url, label, script.version);
-
- }
-
-
- const foundScriptList = async (scriptList) => {
-
- let rid = 0;
- let g = () => {
- if (!scriptList || scriptList.isConnected !== true) return;
-
- const scriptElements = scriptList.querySelectorAll('li[data-script-id]:not([e8kk])');
-
- for (const element of scriptElements) {
- element.setAttribute('e8kk', '1');
-
- const scriptID = +element.getAttribute('data-script-id');
- if (!(scriptID > 0)) continue;
-
- // blacklisted scripts
- if (gmc.get('nonLatins')) hideBlacklistedScript(element, 'nonLatins');
- if (gmc.get('blacklist')) hideBlacklistedScript(element, 'blacklist');
- if (gmc.get('customBlacklist')) hideBlacklistedScript(element, 'customBlacklist');
-
- // hidden scripts
- if (gmc.get('hideHiddenScript')) hideHiddenScript(element, scriptID, true);
-
- // install button
- if (gmc.get('showInstallButton')) {
- showInstallButton(scriptID, element)
- }
- }
-
- }
- let f = (entries) => {
- const tid = ++rid
- if (entries && entries.length) requestAnimationFrame(() => {
- if (tid === rid) g();
- });
- }
- let mo = new MutationObserver(f);
- mo.observe(scriptList, { subtree: true, childList: true });
-
- g();
-
- }
-
- const onReady = async () => {
- addSettingsToMenu();
-
-
- setTimeout(() => {
- let installBtn = document.querySelector('a[data-script-id][data-script-version]')
- let scriptID = installBtn && installBtn.textContent ? +installBtn.getAttribute('data-script-id') : 0;
- if (scriptID > 0) {
- getScriptData(scriptID, true);
- } else {
-
-
- const userLink = document.querySelector('#site-nav .user-profile-link a[href]');
- let userID = userLink.getAttribute('href');
-
- userID = /users\/(\d+)/.exec(userID);
- if (userID) userID = userID[1];
- if (userID) {
- userID = +userID;
- if (userID > 0) {
- getUserData(userID, true);
- }
- }
-
-
- }
- }, 740);
-
- const userLink = document.querySelector('.user-profile-link a[href]');
- const userID = userLink ? userLink.getAttribute('href') : undefined;
-
- // blacklisted scripts / hidden scripts / install button
- if (window.location.pathname !== userID && !/discussions/.test(window.location.pathname) && (gmc.get('hideBlacklistedScripts') || gmc.get('hideHiddenScript') || gmc.get('showInstallButton'))) {
-
- const scriptList = document.querySelector('.script-list');
- if (scriptList) {
- foundScriptList(scriptList);
- } else {
- const timeout = Date.now() + 3000;
- /** @type {MutationObserver | null} */
- let mo = null;
- const mutationCallbackForScriptList = () => {
- if (!mo) return;
- const scriptList = document.querySelector('.script-list');
- if (scriptList) {
- mo.disconnect();
- mo.takeRecords();
- mo = null;
- foundScriptList(scriptList);
- } else if (Date.now() > timeout) {
- mo.disconnect();
- mo.takeRecords();
- mo = null;
- }
- }
- mo = new MutationObserver(mutationCallbackForScriptList);
- mo.observe(document, { subtree: true, childList: true });
- }
-
-
- // hidden scripts on details page
- if (gmc.get('hideHiddenScript') && document.querySelector('#script-info') && document.querySelector('#script-info .install-link[data-script-id]')) {
- const id = +document.querySelector('#script-info .install-link[data-script-id]').getAttribute('data-script-id');
- hideHiddenScript(document.querySelector('#script-info'), id, false);
- }
-
- // add options and style for blacklisted/hidden scripts
- if (gmc.get('hideBlacklistedScripts') || gmc.get('hideHiddenScript')) {
- addOptions();
- UU.addStyle(mWindow.pageCSS);
- }
- }
-
- // total installs
- if (gmc.get('showTotalInstalls') && document.querySelector('#user-script-list')) {
- const dailyInstalls = [];
- const totalInstalls = [];
-
- const dailyInstallElements = document.querySelectorAll('#user-script-list li dd.script-list-daily-installs');
- for (const element of dailyInstallElements) {
- dailyInstalls.push(parseInt(element.textContent.replace(/\D/g, ''), 10));
- }
-
- const totalInstallElements = document.querySelectorAll('#user-script-list li dd.script-list-total-installs');
- for (const element of totalInstallElements) {
- totalInstalls.push(parseInt(element.textContent.replace(/\D/g, ''), 10));
- }
-
- const dailyInstallsSum = dailyInstalls.reduce((a, b) => a + b, 0);
- const totalInstallsSum = totalInstalls.reduce((a, b) => a + b, 0);
-
- const convertLi = (li) => {
-
- if (!li) return null;
- const a = li.firstElementChild
- if (a === null) return li;
- if (a === li.lastElementChild && a.nodeName === 'A') return a;
-
-
- return null;
- }
-
- const dailyOption = convertLi(document.querySelector('#script-list-sort .list-option:nth-child(1)'));
- dailyOption && dailyOption.insertAdjacentHTML('beforeend', `<span> (${dailyInstallsSum.toLocaleString()})</span>`);
-
- const totalOption = convertLi(document.querySelector('#script-list-sort .list-option:nth-child(2)'));
- totalOption && totalOption.insertAdjacentHTML('beforeend', `<span> (${totalInstallsSum.toLocaleString()})</span>`);
- }
-
- // milestone notification
- if (gmc.get('milestoneNotification')) {
- const milestones = gmc.get('milestoneNotification').replace(/\s/g, '').split(',').map(Number);
-
- if (!userID) return;
-
- const userData = await getUserData(+userID.match(/\d+(?=\D)/g));
- if (!userData) return;
-
- const [totalInstalls, lastMilestone] = await Promise.all([
- getTotalInstalls(userData),
- GM.getValue('lastMilestone', 0)]);
-
- const milestone = milestones.filter(milestone => totalInstalls >= milestone).pop();
-
- UU.log(`total installs are "${totalInstalls}", milestone reached is "${milestone}", last milestone reached is "${lastMilestone}"`);
-
- if (milestone <= lastMilestone) return;
-
- if (milestone && milestone >= 0) {
-
-
- GM.setValue('lastMilestone', milestone);
-
- const lang = document.documentElement.lang;
- const text = (locales[lang] ? locales[lang].milestone : locales.en.milestone).replace('$1', milestone.toLocaleString());
-
- if (GM.info.scriptHandler !== 'Userscripts' && typeof GM.notification === 'function') {
- GM.notification({
- text,
- title: GM.info.script.name,
- image: logo,
- onclick: () => {
- window.location = `https://${window.location.hostname}${userID}#user-script-list-section`;
- }
- });
- } else {
- UU.alert(text);
- }
-
- }
-
- }
- }
-
-
-
- Promise.resolve().then(() => {
- if (document.readyState !== 'loading') {
- onReady();
- } else {
- window.addEventListener("DOMContentLoaded", onReady, false);
- }
- });
-
- })();