// ==UserScript== // @name Greasy Fork++ // @author CY Fung & Davide // @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.1.0 // @require https://fastly.jsdelivr.net/gh/sizzlemctwizzle/GM_config@06f2015c04db3aaab9717298394ca4f025802873/gm_config.min.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 content // @downloadURL none // ==/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:
Choose which lists to activate in the section below, press Ctrl + Alt + B to show Blacklisted scripts', section: ['Features'], labelPos: 'right', type: 'checkbox', default: true }, hideHiddenScript: { label: 'Hide scripts:
Add a button to hide the script
See and edit the list of hidden scripts below, press Ctrl + Alt + H to show Hidden script', labelPos: 'right', type: 'checkbox', default: true }, showInstallButton: { label: 'Install button:
Add to the scripts list a button to install the script directly', labelPos: 'right', type: 'checkbox', default: true }, showTotalInstalls: { label: 'Installations:
Shows the number of daily and total installations on the user profile', labelPos: 'right', type: 'checkbox', default: true }, milestoneNotification: { label: 'Milestone notifications:
Get notified whenever your total installs got over any of these milestone
Separate milestones with a comma, leave blank to turn off notifications
', 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:
This list blocks all scripts with non-Latin characters in the title/description', section: ['Lists'], labelPos: 'right', type: 'checkbox', default: false // not true }, blacklist: { label: 'Blacklist:
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"', labelPos: 'right', type: 'checkbox', default: true }, customBlacklist: { label: 'Custom Blacklist:
Personal blacklist defined by a set of unwanted words
Separate unwanted words with a comma (example: YouTube, Facebook, pizza), leave blank to disable this list
', labelPos: 'left', type: 'text', title: 'Separate unwanted words with a comma!', size: 150, default: '' }, hiddenList: { label: 'Hidden Scripts:
Block individual undesired scripts by their unique IDs
Separate IDs with a comma
', 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 label::before { content:''; display:block; position:absolute; left:0; right:0; top:0; bottom:0; z-index:1; } #greasyfork-plus label { position:relative; z-index:0; } */ html { color: #222; background: #f9f9f9; } #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[class] { 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% } body > #greasyfork-plus_wrapper:only-child { 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_wrapper > #greasyfork-plus_buttons_holder:last-child { position: fixed; bottom: 0; right: 0; margin: 0 12px 6px 0; } #greasyfork-plus .saveclose_buttons[class] { padding: 4px 14px; margin: 6px; } #greasyfork-plus .section_header_holder#greasyfork-plus_section_2[class] { position: fixed; left: 0; bottom: 0; margin: 8px; } #greasyfork-plus .section_header#greasyfork-plus_section_header_2[class] { background: #000; color: #eee; } #greasyfork-plus_header[class]{ font-size: 16pt; font-weight: bold; } `; 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; margin: 0 8px; } [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)) } const isScriptFirstUse = await GM.getValue('firstUse', true); 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); } }, /** @param {Document} document */ 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)); } document.body.addEventListener('mousedown', (event)=>{ if (event.detail > 1 && !event.ctrlKey && !event.altKey && !event.metaKey && !event.shiftKey && !event.defaultPrevented) { event.preventDefault(); event.stopPropagation(); event.stopImmediatePropagation(); } }, true); }, 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(); if (typeof GM.registerMenuCommand === 'function') { GM.registerMenuCommand('Configure', () => gmc.open()); GM.registerMenuCommand('Reset Everything', () => { Promise.all([ GM.deleteValue('hiddenList'), GM.deleteValue('lastMilestone'), GM.deleteValue('firstUse') ]).then(() => { setTimeout(() => window.location.reload(false), 50); }) }); } UU.init({ id, logging: gmc.get('logging') }); UU.log(nonLatins); UU.log(blacklist); UU.log(hiddenList); const _VM = (typeof VM !== 'undefined' ? VM : null) || { shortcut: { register: () => { } } }; let avoidDuplication = 0; const avoidDuplicationF = () => { const p = avoidDuplication; avoidDuplication = Date.now(); if (avoidDuplication - p < 30) return false; return true; } const shortcuts = [ ['ctrlcmd-alt-s', () => avoidDuplicationF() && gmc.open()], ['ctrlcmd-alt-ß', () => avoidDuplicationF() && gmc.open()], ['ctrlcmd-alt-b', () => avoidDuplicationF() && toggleListDisplayingItem('blacklisted')], ['ctrlcmd-alt-∫', () => avoidDuplicationF() && toggleListDisplayingItem('blacklisted')], ['ctrlcmd-alt-h', () => avoidDuplicationF() && toggleListDisplayingItem('hidden')], ['ctrlcmd-alt-˙', () => avoidDuplicationF() && toggleListDisplayingItem('hidden')] ] for (const [scKey, scFn] of shortcuts) { _VM.shortcut.register(scKey, scFn); } const addSettingsToMenu = () => { const nav = document.querySelector('#site-nav > nav') if (!nav) return; const scriptName = GM.info.script.name; const scriptVersion = GM.info.script.version; const menu = document.createElement('li'); menu.classList.add(id); menu.setAttribute('alt', `${scriptName} ${scriptVersion}`); menu.setAttribute('title', `${scriptName} ${scriptVersion}`); const link = document.createElement('a'); link.setAttribute('href', '#'); link.textContent = GM.info.script.name; menu.appendChild(link); nav.insertBefore(menu, document.querySelector('#site-nav > nav > li:first-of-type')); 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 = `
${GM.info.script.name} Lists:
`; 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 = `${blockLabel(isHidden)}`; const blockButtonHeaderHTML = `${blockLabel(isHidden)}`; 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', `${label} ${version}`); }; 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 ? userLink.getAttribute('href') : ''; userID = userID ? /users\/(\d+)/.exec(userID) : null; 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 (userID && 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 const installLinkElement = document.querySelector('#script-info .install-link[data-script-id]'); if (gmc.get('hideHiddenScript') && installLinkElement) { const id = +installLinkElement.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', ` (${dailyInstallsSum.toLocaleString()})`); const totalOption = convertLi(document.querySelector('#script-list-sort .list-option:nth-child(2)')); totalOption && totalOption.insertAdjacentHTML('beforeend', ` (${totalInstallsSum.toLocaleString()})`); } // 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 (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); } } } if(isScriptFirstUse) GM.setValue('firstUse', false).then(()=>{ gmc.open(); }); } Promise.resolve().then(() => { if (document.readyState !== 'loading') { onReady(); } else { window.addEventListener("DOMContentLoaded", onReady, false); } }); })();