// ==UserScript== // @name Magic Userscript+ : Show Site All UserJS // @name:zh Magic Userscript+ : 显示当前网站所有可用的UserJS脚本 Jaeger // @name:zh-CN Magic Userscript+ : 显示当前网站所有可用的UserJS脚本 Jaeger // @name:zh-TW Magic Userscript+ : 顯示當前網站所有可用的UserJS腳本 Jaeger // @name:ja Magic Userscript+ : 現在のサイトの利用可能なすべてのUserJSスクリプトを表示するJaeger // @name:ru-RU Magic Userscript+ : Показать пользовательские скрипты (UserJS) для сайта. Jaeger // @name:ru Magic Userscript+ : Показать пользовательские скрипты (UserJS) для сайта. Jaeger // @description Show current site all UserJS, the easier way to install UserJs for Tampermonkey. // @description:zh 显示当前网站的所有可用UserJS(Tampermonkey)脚本,交流QQ群:104267383 // @description:zh-CN 显示当前网站的所有可用UserJS(Tampermonkey)脚本,交流QQ群:104267383 // @description:zh-TW 顯示當前網站的所有可用UserJS(Tampermonkey)腳本,交流QQ群:104267383 // @description:ja 現在のサイトで利用可能なすべてのUserJS(Tampermonkey)スクリプトを表示します。 // @description:ru-RU Показывает пользовательские скрипты (UserJS) для сайта. Легкий способ установить пользовательские скрипты для Tampermonkey. // @description:ru Показывает пользовательские скрипты (UserJS) для сайта. Легкий способ установить пользовательские скрипты для Tampermonkey. // @author Magic // @version 3.7.15 // @icon  // @supportURL https://github.com/magicoflolis/Userscript-Plus/issues/new // @namespace https://github.com/magicoflolis/Userscript-Plus // @homepageURL https://github.com/magicoflolis/Userscript-Plus // @license MIT // @connect greasyfork.org // @connect sleazyfork.org // @connect github.com // @connect openuserjs.org // @connect cdn.jsdelivr.net // @match https://*/* // @exclude http*://*stripe.com/* // @exclude http*://*ica.yandex.com/* // @exclude http*://*skrill.com/* // @exclude http*://*zalo.me/* // @exclude http*://*.bluesnap.com/* // @grant GM_xmlhttpRequest // @grant GM_openInTab // @grant GM_getValue // @grant GM_setValue // @grant GM_info // @compatible chrome // @compatible firefox // @compatible edge // @compatible opera // @compatible safari // @noframes // @run-at document-end // @downloadURL none // ==/UserScript== /** * Injected stylesheet * https://github.com/magicoflolis/Userscript-Plus/tree/master/userscript/src/sass */ const main_css = `magic-userjs{cursor:default}magic-userjs *{line-height:normal}.hidden{display:none !important;z-index:-1 !important}.main{position:fixed;height:492px;margin-left:1rem;margin-right:1rem;bottom:1rem;right:1rem;background:#495060 !important;border:1px solid rgba(0,0,0,0);border-radius:10px;font-size:14px !important;font-family:arial,sans-serif !important}.main:not(.hidden){z-index:100000000000000020 !important;display:grid !important}.main *:not(magicuserjs-a,magicuserjs-btn,count-frame,.counterframe,.count,label,.magicuserjs-switch){background:#495060 !important}.main *:not(magicuserjs-a,magicuserjs-btn,count-frame,.counterframe,.count){color:#fff !important}.counterframe{border-radius:100%;padding:.5%;width:25px !important;height:25px !important}.count{background:rgba(0,0,0,0)}.mainframe{background:#495060 !important;color:#fff !important;border:2px solid rgba(0,0,0,0);border-radius:100%;padding:.5% !important;position:fixed;bottom:1rem;right:1rem;width:25px !important;height:25px !important}.mainframe:not(.hidden){z-index:100000000000000020 !important;display:block}count-frame{border:2px solid rgba(0,0,0,0);font-size:16px;font-weight:400;display:block;text-align:center}count-frame{line-height:normal !important;width:auto !important;height:auto !important}.magicuserjs-header{display:flex;gap:10px;border-bottom:1px solid #fff;border-top-left-radius:10px;border-top-right-radius:10px;height:fit-content;padding:10px;font-size:1em;justify-content:space-between}.magicuserjs-body{overflow-y:scroll;overflow-x:hidden}.magicuserjs-cfg,.magicuserjs-body{scrollbar-color:#fff #2e323d;scrollbar-width:thin;border:1px solid rgba(0,0,0,0);border-bottom-left-radius:10px;border-bottom-right-radius:10px}.magicuserjs-cfg{height:fit-content}.magicuserjs-cfg mujs-section>label{display:flex;justify-content:space-between;padding:.5em}.magicuserjs-cfg mujs-section>label input[type*=number]{position:relative;border-radius:4px;border:1px solid #fff}.magicuserjs-cfg .magicuserjs-inlab{position:relative;width:38px;border-radius:20px;margin:5px}.magicuserjs-cfg .magicuserjs-inlab input[type*=checkbox]{display:none}.magicuserjs-cfg .magicuserjs-inlab input[type*=checkbox]:checked+label{margin-left:0;background-color:rgba(255,255,255,.568)}.magicuserjs-cfg .magicuserjs-inlab input[type*=checkbox]:checked+label:before{right:0px}.magicuserjs-cfg .magicuserjs-inlab input[type*=checkbox]#greasyfork:checked+label,.magicuserjs-cfg .magicuserjs-inlab input[type*=checkbox]#sleazyfork:checked+label{background-color:rgba(0,183,255,.568)}.magicuserjs-cfg .magicuserjs-inlab input[type*=checkbox]#openuserjs:checked+label{background-color:rgba(237,63,20,.568)}.magicuserjs-cfg .magicuserjs-inlab input[type*=checkbox]#github:checked+label{background-color:rgba(36,41,47,.568)}.magicuserjs-cfg .magicuserjs-inlab label{display:block;overflow:hidden;height:16px;padding:0;border-radius:20px;border:1px solid #fff;background-color:#495060}.magicuserjs-cfg .magicuserjs-inlab label:before{content:"";display:block;width:20px;height:20px;margin:-2px;background:#fff;position:absolute;top:0;right:20px;border-radius:20px}.magicuserjs-eframe,.frame{display:grid;grid-auto-flow:column;grid-auto-columns:1fr;font-size:1em}.frame{border-bottom:1px solid #fff}@media screen and (max-width: 800px){.frame{display:flow-root !important;height:fit-content !important}}.frame:not(.sf) magicuserjs-a{color:#00b7ff !important}.frame:not(.sf) magicuserjs-btn{color:#fff;background-color:#2d8cf0;border-color:#2d8cf0}.frame.sf magicuserjs-a{color:#e75531 !important}.frame.sf magicuserjs-btn{background-color:#ed3f14 !important;border-color:#ed3f14 !important}.magicuserjs-name{font-size:inherit !important;display:grid;margin-left:1%;margin-top:.67em;margin-bottom:.67em}.magicuserjs-name span{font-size:.8em !important}.magicuserjs-eframe{margin-right:1%;margin-top:.67em;margin-bottom:.67em}@media screen and (max-width: 800px){.magicuserjs-eframe{height:fit-content !important}}.magicuserjs-uframe{display:grid;grid-auto-flow:column;grid-gap:1em}mujs-btn{font-style:normal;font-weight:400;font-variant:normal;text-transform:none;text-rendering:auto;border:1px solid #fff;font-size:16px;border-radius:4px;line-height:1;padding:6px 15px}mujs-column{display:flex;gap:10px}@media screen and (max-width: 800px){mujs-column{flex-flow:row wrap}mujs-column.btnframe{flex-flow:row wrap-reverse !important}}mujs-row{display:flex;gap:10px;flex-direction:column}mujs-btn svg{fill:#fff;width:14px;height:14px}input[type*=number],input[type*=text]{border:rgba(0,0,0,0);outline:none !important}magicuserjs-btn{font-size:12px;border-radius:3px;font-style:normal;padding:6px 15%;font-weight:400;font-variant:normal;line-height:21px}magicuserjs-a,magicuserjs-btn,.magicuserjs-cfg *:not(input[type*=text],input[type*=number]),.mainbtn,.mainframe,mujs-btn{cursor:pointer !important}.magicuserjs-cfg *:not(input[type*=text],input[type*=number]){user-select:none !important}mujs-btn,input,.magicuserjs-uframe,.magicuserjs-list,.install,.magicuserjs-homepage{width:fit-content;height:fit-content}`; (() => { const log = (...msg) => console.log('[%cUserJS%c] %cDBG', 'color: rgb(29, 155, 240);', '', 'color: rgb(255, 212, 0);', ...msg), /** Information handling for UserJS */ info = (...msg) => console.info('[%cUserJS%c] %cINF', 'color: rgb(29, 155, 240);', '', 'color: rgb(0, 186, 124);', ...msg), err = (...msg) => console.error('[%cUserJS%c] %cERROR', 'color: rgb(29, 155, 240);', '', 'color: rgb(249, 24, 128);', ...msg); let checkGMSupport = !Object.is(GM,undefined), MU = { getValue: false, info: { script: { version: 'Bookmarklet' } }, openInTab: false, setValue: false, xmlhttpRequest: false, }; if(checkGMSupport) { MU = { getValue: GM_getValue, info: GM_info, openInTab: GM_openInTab, setValue: GM_setValue, xmlhttpRequest: GM_xmlhttpRequest, }; }; let langs = { en: { daily: 'Daily Installs', close: 'No longer show', search: 'Search', searcher: 'Title | Description | Author...', install: 'Install', issue: 'New Issue', version: 'Version', updated: 'Last Updated', legacy: 'Legacy', total: 'Total Installs', rating: 'Ratings', good: 'Good', ok: 'Ok', bad: 'Bad', created: 'Created', redirect: 'Greasy Fork for adults', filter: 'Filter out other languages', dtime: 'Display Timeout', save: 'Save', }, es: { daily: 'Instalaciones diarias', close: 'Ya no se muestra', search: 'Busque en', searcher: 'Título | Descripción | Autor...', install: 'Instalar', issue: 'Nueva edición', version: 'Versión', updated: 'Última actualización', legacy: 'Legado', total: 'Total de instalaciones', rating: 'Clasificaciones', good: 'Bueno', ok: 'Ok', bad: 'Malo', created: 'Creado', redirect: 'Greasy Fork para adultos', filter: 'Filtrar otros idiomas', dtime: 'Mostrar el tiempo de espera', save: 'Guardar', }, ru: { daily: 'Ежедневные установки', close: 'Больше не показывать', search: 'Поиск', searcher: 'Название | Описание | Автор...', install: 'Установите', issue: 'Новый выпуск', version: 'Версия', updated: 'Последнее обновление', legacy: 'Наследие', total: 'Всего установок', rating: 'Рейтинги', good: 'Хорошо', ok: 'Хорошо', bad: 'Плохо', created: 'Создано', redirect: 'Greasy Fork для взрослых', filter: 'Отфильтровать другие языки', dtime: 'Тайм-аут отображения', save: 'Сохранить', }, ja: { daily: 'デイリーインストール', close: '表示されなくなりました', search: '検索', searcher: 'タイトル|説明|著者...', install: 'インストール', issue: '新刊のご案内', version: 'バージョン', updated: '最終更新日', legacy: 'レガシー', total: '総インストール数', rating: 'レーティング', good: 'グッド', ok: '良い', bad: '悪い', created: '作成', redirect: '大人のGreasyfork', filter: '他の言語をフィルタリングする', dtime: '表示タイムアウト', save: '拯救', }, fr: { daily: 'Installations quotidiennes', close: 'Ne plus montrer', search: 'Recherche', searcher: 'Titre | Description | Auteur...', install: 'Installer', issue: 'Nouveau numéro', version: 'Version', updated: 'Dernière mise à jour', legacy: 'Héritage', total: 'Total des installations', rating: 'Notations', good: 'Bon', ok: 'Ok', bad: 'Mauvais', created: 'Créé', redirect: 'Greasy Fork pour les adultes', filter: 'Filtrer les autres langues', dtime: `Délai d'affichage`, save: 'Sauvez', }, zh: { daily: '日常安装', close: '不再显示', search: '搜索', searcher: '标题|描述|作者...', install: '安装', issue: '新问题', version: '版本', updated: '最后更新', legacy: '遗产', total: '总安装量', rating: '评级', good: '好的', ok: '好的', bad: '不好', created: '创建', redirect: '大人的Greasyfork', filter: '过滤掉其他语言', dtime: '显示超时', save: '拯救', }, }, defcfg = { cache: true, lang: langs[navigator.language.split('-')[0] ?? 'en'], filterlang: false, sleazyredirct: false, time: 10000, blacklist: [ { enabled: true, regex: true, flags: '', name: 'Default 1', url: '(gov|cart|checkout|login|join|signin|signup|sign-up|password|reset|password_reset)', }, { enabled: true, regex: true, flags: '', name: 'Default 2', url: '(pay|bank|money|localhost|authorize|checkout|bill|wallet|router)', }, ], engines: [ { enabled: true, name: 'greasyfork', url: 'https://greasyfork.org/scripts/by-site', }, { enabled: true, name: 'sleazyfork', url: 'https://sleazyfork.org/scripts/by-site', }, { enabled: false, name: 'openuserjs', url: 'https://openuserjs.org/?q=', }, { enabled: false, name: 'github', url: 'https://github.com/search?l=JavaScript&o=desc&q="==UserScript=="+', }, { enabled: false, name: 'gist', url: 'https://gist.github.com/search?l=JavaScript&o=desc&q="==UserScript=="+', }, ] }, cfg = {}, sitegfcount = 0, sitesfcount = 0; function main() { const win = self ?? window, doc = win.document, estr = str => Object.is(str,null) || Object.is(str,undefined) || typeof str === 'string' && Object.is(str.trim(),''), save = () => { try { if(checkGMSupport) { if(!cfg.cache) { localStorage.setItem('MUJSConfig',JSON.stringify(cfg)); } else { MU.setValue('Config',JSON.stringify(cfg)); }; } else { localStorage.setItem('MUJSConfig',JSON.stringify(cfg)); }; log('Saved:',cfg); } catch(e) {err(e)}; }, /** Can create various elements */ make = (element,cname,attrs = {}) => { let el = doc.createElement(element); if(!estr(cname)) { el.className = cname; }; if(attrs) { for(let key in attrs) { el[key] = attrs[key] } }; return el; }, ifram = make('iframe','', { src: 'about:blank', style: `position: fixed; bottom: 1rem; right: 1rem; height: 525px; width: 90%; margin-left: 1rem; margin-right: 1rem; z-index: 100000000000000020 !important; ` }), container = make('magic-userjs'); if(estr(container.attachShadow)) { doc.body.append(ifram); ifram.onload = () => { ifram.contentDocument.body.append(container); if(estr(container.attachShadow)) { ifram.contentDocument.body.setAttribute('style','background-color: black;color: white;'); ifram.contentDocument.body.innerHTML = '[ERROR] Unsupported { attachShadow }... yeah still need to work on that :)'; delay(5000).then(() => ifram.remove()); } else { container.attachShadow({ mode: 'open' }); }; }; } else { container.attachShadow({ mode: 'open' }); doc.body.append(container); }; const qs = (element, selector) => { selector = selector ?? doc ?? doc.body; return selector.querySelector(element); }, qsA = (element, selector) => { selector = selector ?? doc ?? doc.body; return selector.querySelectorAll(element); }, /** Waits until element exists */ query = async (element, selector) => { selector = selector ?? doc ?? doc.body; while(estr(selector.querySelector(element))) { await new Promise(resolve => requestAnimationFrame(resolve) ) }; return selector.querySelector(element); }, sh = element => container.shadowRoot.querySelector(element), shA = element => container.shadowRoot.querySelectorAll(element), delay = ms => new Promise(resolve => setTimeout(resolve, ms)), clk = e => e.dispatchEvent(new MouseEvent('click')), isMobile = () => { let a = navigator.userAgent || win.opera; return /(android|bb\d+|meego).+mobile|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|hiptop|iemobile|ip(hone|od|ad)|iris|kindle|lge |maemo|midp|mmp|mobile.+firefox|netfront|opera m(ob|in)i|palm( os)?|phone|p(ixi|re)\/|plucker|pocket|psp|series(4|6)0|symbian|treo|up\.(browser|link)|vodafone|wap|windows (ce|phone)|xda|xiino/i.test(a)||/1207|6310|6590|3gso|4thp|50[1-6]i|770s|802s|a wa|abac|ac(er|oo|s-)|ai(ko|rn)|al(av|ca|co)|amoi|an(ex|ny|yw)|aptu|ar(ch|go)|as(te|us)|attw|au(di|-m|r |s )|avan|be(ck|ll|nq)|bi(lb|rd)|bl(ac|az)|br(e|v)w|bumb|bw-(n|u)|c55\/|capi|ccwa|cdm-|cell|chtm|cldc|cmd-|co(mp|nd)|craw|da(it|ll|ng)|dbte|dc-s|devi|dica|dmob|do(c|p)o|ds(12|-d)|el(49|ai)|em(l2|ul)|er(ic|k0)|esl8|ez([4-7]0|os|wa|ze)|fetc|fly(-|_)|g1 u|g560|gene|gf-5|g-mo|go(\.w|od)|gr(ad|un)|haie|hcit|hd-(m|p|t)|hei-|hi(pt|ta)|hp( i|ip)|hs-c|ht(c(-| |_|a|g|p|s|t)|tp)|hu(aw|tc)|i-(20|go|ma)|i230|iac( |-|\/)|ibro|idea|ig01|ikom|im1k|inno|ipaq|iris|ja(t|v)a|jbro|jemu|jigs|kddi|keji|kgt( |\/)|klon|kpt |kwc-|kyo(c|k)|le(no|xi)|lg( g|\/(k|l|u)|50|54|-[a-w])|libw|lynx|m1-w|m3ga|m50\/|ma(te|ui|xo)|mc(01|21|ca)|m-cr|me(rc|ri)|mi(o8|oa|ts)|mmef|mo(01|02|bi|de|do|t(-| |o|v)|zz)|mt(50|p1|v )|mwbp|mywa|n10[0-2]|n20[2-3]|n30(0|2)|n50(0|2|5)|n7(0(0|1)|10)|ne((c|m)-|on|tf|wf|wg|wt)|nok(6|i)|nzph|o2im|op(ti|wv)|oran|owg1|p800|pan(a|d|t)|pdxg|pg(13|-([1-8]|c))|phil|pire|pl(ay|uc)|pn-2|po(ck|rt|se)|prox|psio|pt-g|qa-a|qc(07|12|21|32|60|-[2-7]|i-)|qtek|r380|r600|raks|rim9|ro(ve|zo)|s55\/|sa(ge|ma|mm|ms|ny|va)|sc(01|h-|oo|p-)|sdk\/|se(c(-|0|1)|47|mc|nd|ri)|sgh-|shar|sie(-|m)|sk-0|sl(45|id)|sm(al|ar|b3|it|t5)|so(ft|ny)|sp(01|h-|v-|v )|sy(01|mb)|t2(18|50)|t6(00|10|18)|ta(gt|lk)|tcl-|tdg-|tel(i|m)|tim-|t-mo|to(pl|sh)|ts(70|m-|m3|m5)|tx-9|up(\.b|g1|si)|utst|v400|v750|veri|vi(rg|te)|vk(40|5[0-3]|-v)|vm40|voda|vulc|vx(52|53|60|61|70|80|81|83|85|98)|w3c(-| )|webc|whit|wi(g |nc|nw)|wmlb|wonu|x700|yas-|your|zeto|zte-/i.test(a.substr(0,4)); }, halt= (e) => { e.preventDefault(); e.stopPropagation(); }, ael = (elm, event, callback) => { try { elm = elm ?? doc; if(isMobile()) { if(event === 'click') { event = 'mouseup'; // elm.addEventListener('mouseup', callback); elm.addEventListener('touchstart', callback); elm.addEventListener('touchend', callback); }; }; if(event === 'fclick') {event = 'click'}; return elm.addEventListener(event, callback); } catch(error) { err(error); }; }, pbar = (e) => { if(e.total === 0) { return info(`Loaded: ${e.loaded}`) }; return info(`Progress: ${e.loaded / e.total}%`) }, fetchURL = async (url,method = 'GET',responseType = 'json',params = {}, forcefetch) => { try { return new Promise((resolve, reject) => { if(checkGMSupport && !forcefetch) { MU.xmlhttpRequest({ method: method, url, responseType, ...params, onprogress: pbar, onerror: e => reject(e), onload: (r) => { if(r.status !== 200) reject(`${r.status} ${url}`); resolve(r.response); }, }); } else { fetch(url, { method: method, ...params, }).then((response) => { if(!response.ok) reject(response); if(responseType.match(/json/gi)) { resolve(response.json()); } else if(responseType.match(/text/gi)) { resolve(response.text()); } else if(responseType.match(/blob/gi)) { resolve(response.blob()); }; resolve(response); }); } }); } catch (error) {err(error)} }, openpage = (url) => { if(checkGMSupport) { return MU.openInTab(url, { active: true, insert: true, }); }; let dwnbtn = make('a','magicuserjs-dwnbtn', { href: url, target: '_blank', rel: 'noopener', }); clk(dwnbtn); return dwnbtn.remove(); }, createjs = (ujs, issleazy) => { let frame = make('magic-userjs',`frame ${issleazy ? 'sf' : ''}`), fname = make('magic-userjs','magicuserjs-name'), ftitle = make('magicuserjs-a','magicuserjs-homepage', { title: ujs.name, innerHTML: ujs.name, onclick: (e) => { e.preventDefault(); openpage(ujs.url); } }), fver = make('magic-userjs','magicuserjs-list', { innerHTML: `${cfg.lang.version}: ${ujs.version}`, }), fcreated = make('magic-userjs','magicuserjs-list', { innerHTML: `${cfg.lang.created}: ${new Intl.DateTimeFormat(navigator.language).format(new Date(ujs.created_at))}`, }), fmore = make('mujs-column','magicuserjs-list hidden', { style: 'margin-top: 3px;', }), ftotal = make('magic-userjs','magicuserjs-list', { innerHTML: `${cfg.lang.total}: ${ujs.total_installs}`, }), fgood = make('magic-userjs','magicuserjs-list', { title: cfg.lang.good, innerHTML: `${cfg.lang.rating}: ${ujs.good_ratings}`, }), fok = make('magic-userjs','magicuserjs-list', { title: cfg.lang.ok, innerHTML: ujs.ok_ratings, }), fbad = make('magic-userjs','magicuserjs-list', { title: cfg.lang.bad, innerHTML: ujs.bad_ratings, }), fdesc = make('magic-userjs','magicuserjs-list', { style: 'cursor: pointer; margin-top: 3px;', title: ujs.description, innerHTML: ujs.description, onclick: (e) => { e.preventDefault(); if(fmore.classList.contains('hidden')) { fmore.classList.remove('hidden'); } else { fmore.classList.add('hidden'); } }, }), eframe = make('magic-userjs','magicuserjs-eframe'), uframe = make('magic-userjs','magicuserjs-uframe'), fdaily = make('magic-userjs','magicuserjs-list', { title: cfg.lang.daily, innerHTML: ujs.daily_installs, }), fupdated = make('magic-userjs','magicuserjs-list', { title: cfg.lang.updated, innerHTML: new Intl.DateTimeFormat(navigator.language).format(new Date(ujs.code_updated_at)), }), fdwn = make('magicuserjs-btn','install', { title: `${cfg.lang.install} '${ujs.name}'`, innerHTML: cfg.lang.install, onclick: (e) => { e.preventDefault(); openpage(ujs.code_url); }, }); for(let u of ujs.users) { let user = make('magicuserjs-a','magicuserjs-euser', { innerHTML: u.name, onclick: (e) => { e.preventDefault(); openpage(u.url); }, }); uframe.append(user); }; eframe.append(uframe,fdaily,fupdated,fdwn); fmore.append(ftotal,fgood,fok,fbad,fver,fcreated); fname.append(ftitle,fdesc,fmore); frame.append(fname,eframe); sh('.magicuserjs-body').append(frame); }, countsite = () => { try { if(/greasyfork\.org/.test(doc.location.hostname) && cfg.sleazyredirct) { let otherSite = /greasyfork\.org/.test(document.location.hostname) ? 'sleazyfork' : 'greasyfork'; qs('span.sign-in-link') ? /scripts\/\d+/.test(document.location.href) ? !qs('#script-info') && (otherSite == 'greasyfork' || qs('div.width-constraint>section>p>a')) ? location.href = location.href.replace(/\/\/([^.]+\.)?(greasyfork|sleazyfork)\.org/, '//$1' + otherSite + '.org') : false : false : false; }; if(!container.attachShadow) return false; let host = location.hostname.split('.').splice(-2).join('.'), rebuild = false, siteujs = [], main = make('magic-userjs','main hidden'), usercss = make('style', 'primary-stylesheet', {innerHTML: main_css,}), tbody = make('magic-userjs','magicuserjs-body'), header = make('magic-userjs','magicuserjs-header'), cfgpage = make('mujs-row','magicuserjs-cfg hidden', { innerHTML: ` ` }), countframe = make('mujs-column'), gfcountframe = make('magic-userjs', 'counterframe', { style: 'background: #00b7ff;' }), sfcountframe = make('magic-userjs', 'counterframe', { style: 'background: #ed3f14;' }), gfcounter = make('count-frame','count', { title: 'https://greasyfork.org + https://sleazyfork.org', }), sfcounter = make('count-frame','count', { title: 'https://openuserjs.org', }), buildlist = async () => { const template = { bad_ratings: 0, good_ratings: 0, ok_ratings: 0, daily_installs: 0, total_installs: 0, name: 'Not found', description: 'Not found', version: '0.0.0', url: 'about:blank', code_url: 'about:blank', created_at: Date.now(), code_updated_at: Date.now(), users: [{ name: '', url: '', }] }; siteujs = []; sitegfcount = 0; sitesfcount = 0; tbody.innerHTML = ''; gfcounter.innerHTML = sitegfcount; sfcounter.innerHTML = sitesfcount; mainbtn.innerHTML = sitegfcount; let sites = [], custom = []; for(let i of cfg.engines) { if(i.enabled) { if(i.url.match(/fork.org/gi)) { sites.push(fetchURL(`${i.url}/${host}.json`).catch(err),); }; if(i.url.match(/(openuserjs.org|github.com)/gi)) { log(`${i.url}${host}`); custom.push(fetchURL(`${i.url}${host}`,'GET','text').catch(err),); }; }; }; let data = await Promise.all(sites).catch(err); for(let d of data) { for(let ujs of d) { if(ujs.deleted) continue; if(cfg.filterlang && !ujs.locale.includes(navigator.language.split('-')[0] ?? 'en')) continue; siteujs.push( { url: ujs, sleazy: false, }, ); sitegfcount++; }; }; for(let ujs of siteujs) { createjs(ujs.url,ujs.sleazy); }; gfcounter.innerHTML = sitegfcount; mainbtn.innerHTML = sitesfcount + sitegfcount; if(custom.length > 0) { let c = await Promise.all(custom).catch(err), parser = new DOMParser(), htmlDocument = parser.parseFromString(c,'text/html'), selected = htmlDocument.documentElement; if(qs('.col-sm-8 .tr-link',selected)) { for(let i of qsA('.col-sm-8 .tr-link',selected)) { await query('.script-version',i); let fixurl = qs('.tr-link-a',i).href.replaceAll(doc.location.origin,'https://openuserjs.org'), layout = { name: qs('.tr-link-a',i).textContent, description: qs('p',i).textContent, version: qs('.script-version',i).textContent, url: fixurl, code_url: `${fixurl.replaceAll('/scripts','/install')}.user.js`, total_installs: qs('td:nth-child(2) p',i).textContent, created_at: qs('td:nth-child(4) time',i).getAttribute('datetime'), code_updated_at: qs('td:nth-child(4) time',i).getAttribute('datetime'), users: [{ name: qs('.inline-block a',i).textContent, url: qs('.inline-block a',i).href, }] }; for(const key in template) { if(!Object.prototype.hasOwnProperty.call(layout, key)) { layout[key] = template[key]; }; }; createjs(layout, true); sitesfcount++; sfcounter.innerHTML = sitesfcount; }; }; if(qs('.repo-list-item',selected)) { for(let r of qsA('.repo-list-item',selected)) { let layout = {}, fixurl = qs('a',r).href.replaceAll(doc.location.origin,'https://github.com'); layout = Object.assign(layout, { name: qs('a',r).textContent, description: qs('p.mb-1',r).textContent.trim(), url: fixurl, code_url: fixurl, code_updated_at: qs('relative-time.no-wrap',r).getAttribute('datetime'), total_installs: qs('a.Link--muted:nth-child(1)',r) ? qs('a.Link--muted:nth-child(1)',r).textContent : 0, users: [{ name: qs('a',r).href.match(/\/[\w\d-]+\//gi)[0].replaceAll('/',''), url: `https://github.com${qs('a',r).href.match(/\/[\w\d-]+\//gi)}`, }] }); for (const key in template) { if(!Object.prototype.hasOwnProperty.call(layout, key)) { layout[key] = template[key]; }; }; createjs(layout, true); sitesfcount++; sfcounter.innerHTML = sitesfcount; }; }; if(qs('div.gist-snippet',selected)) { for(let g of qsA('div.gist-snippet',selected)) { if(qs('span > a:nth-child(2)',g).textContent.includes('.user.js')) { let layout = {}, fixurl = qs('span > a:nth-child(2)',g).href.replaceAll(doc.location.origin,'https://gist.github.com'); layout = Object.assign(layout, { url: fixurl, code_url: `${fixurl}/raw/${qs('span > a:nth-child(2)',g).textContent}`, created_at: qs('time-ago.no-wrap',g).getAttribute('datetime'), users: [{ name: qs('span > a[data-hovercard-type]',g).textContent, url: qs('span > a[data-hovercard-type]',g).href.replaceAll(doc.location.origin,'https://gist.github.com'), }] }); for(let i of qsA('.file-box table tr .blob-code',g)) { let txt = i.textContent, headers = txt.match(/\/\/\s@[\w][\s\S]+/gi) || []; if(headers.length > 0) { let crop = headers[0].split(/\/\/\s@(name|description|author|version)\s+/gi); if(headers[0].includes('@name') && !headers[0].includes('@namespace')) { layout = Object.assign(layout, { name: crop[2].trim(), }); }; if(headers[0].includes('@description')) { layout = Object.assign(layout, { description: crop[2].trim(), }); }; if(headers[0].includes('@version')) { layout = Object.assign(layout, { version: crop[2].trim(), }); }; } }; for (const key in template) { if(!Object.prototype.hasOwnProperty.call(layout, key)) { layout[key] = template[key]; }; }; createjs(layout, true); sitesfcount++; sfcounter.innerHTML = sitesfcount; }; }; }; sfcounter.innerHTML = sitesfcount; mainbtn.innerHTML = sitesfcount + sitegfcount; }; if(!isNaN(cfg.time)) { delay(cfg.time).then(() => { if(!mainframe.classList.contains('hidden')) { mainframe.classList.add('hidden'); ifram.setAttribute('style','display:none;'); }; }); }; if(Object.is(data[0].length,0) && Object.is(data[1].length,0) && Object.is(custom.length,0)) { tbody.innerHTML = 'No available UserJS for this webpage'; }; }, makecfg = () => { let savebtn = make('mujs-btn', 'save', { style: 'margin: auto;', innerHTML: cfg.lang.save, onclick: async (e) => { halt(e); let t = sh('input#time'); if(t.validity.badInput || t.validity.rangeUnderflow && t.value !== '-1') { t.setAttribute('style','border-radius: 8px; border-width: 2px !important; border-style: solid; border-color: red !important;'); } else { save(); if(rebuild) { buildlist(); rebuild = false; }; if(/greasyfork\.org/.test(doc.location.hostname) && cfg.sleazyredirct) { let otherSite = /greasyfork\.org/.test(document.location.hostname) ? 'sleazyfork' : 'greasyfork'; qs('span.sign-in-link') ? /scripts\/\d+/.test(document.location.href) ? !qs('#script-info') && (otherSite == 'greasyfork' || qs('div.width-constraint>section>p>a')) ? location.href = location.href.replace(/\/\/([^.]+\.)?(greasyfork|sleazyfork)\.org/, '//$1' + otherSite + '.org') : false : false : false; }; } }, }); cfgpage.append(savebtn); for(let i of cfg.engines) { if(sh(`#${i.name}`)) { sh(`#${i.name}`).checked = i.enabled; ael(sh(`#${i.name}`),'change', (e) => { i.enabled = e.target.checked; rebuild = true; }); }; }; sh('input#cache').checked = cfg.cache; ael(sh('input#cache'),'change', (e) => { cfg.cache = e.target.checked; rebuild = true; }); sh('input#sleazyredirct').checked = cfg.sleazyredirct; ael(sh('input#sleazyredirct'),'change', (e) => { cfg.sleazyredirct = e.target.checked; }); ael(sh('input#time'),'beforeinput', (e) => { if(e.target.validity.badInput) { e.target.setAttribute('style','border-radius: 8px; border-width: 2px !important; border-style: solid; border-color: red !important;'); } else { e.target.setAttribute('style',''); } }); ael(sh('input#time'),'input', (e) => { let t = e.target; log(t.value); if(t.validity.badInput || t.validity.rangeUnderflow && t.value !== '-1') { t.setAttribute('style','border-radius: 8px; border-width: 2px !important; border-style: solid; border-color: red !important;'); } else { t.setAttribute('style',''); cfg.time = estr(t.value) ? cfg.time : parseFloat(t.value); } }); sh('input#filter').checked = cfg.filterlang; ael(sh('input#filter'),'change', (e) => { cfg.filterlang = e.target.checked; rebuild = true; }); }, mainframe = make('magic-userjs','mainframe', { onclick: (e) => { e.preventDefault(); main.classList.remove('hidden'); mainframe.classList.add('hidden'); } }), mainbtn = make('count-frame','mainbtn', { innerHTML: '0', }), fsearch = make('mujs-btn','hidden'), searcher = make('input','searcher', { style: 'width: 170px;', autocomplete: 'off', spellcheck: false, type: 'text', placeholder: cfg.lang.searcher, oninput: (e) => { e.preventDefault(); let v = e.target.value; if(!estr(v)) { let reg = new RegExp(v,'gi'); for(let ujs of shA('.frame')) { let m = ujs.children[0], n = ujs.children[1], final = m.textContent.match(reg) || n.textContent.match(reg) || []; if(final.length === 0) { ujs.classList.add('hidden'); } else { ujs.classList.remove('hidden'); }; }; } else { for(let ujs of shA('.frame')) { ujs.classList.remove('hidden') }; }; }, }), searchbtn = make('mujs-btn','search', { title: cfg.lang.search, innerHTML: ``, onclick: (e) => { e.preventDefault(); fsearch.classList.toggle('hidden'); } }), closebtn = make('mujs-btn','close', { title: cfg.lang.close, innerHTML: ``, onclick: (e) => { e.preventDefault(); main.classList.add('hidden'); mainframe.classList.remove('hidden'); if(!isNaN(cfg.time)) { delay(cfg.time).then(() => { mainframe.classList.add('hidden'); ifram.setAttribute('style','display:none;'); }) }; } }), btnframe = make('mujs-column', 'btnframe'), btncfg = make('mujs-btn','settings', { title: 'Settings', innerHTML: ` `, onclick: (e) => { e.preventDefault(); if(cfgpage.classList.contains('hidden')) { cfgpage.classList.remove('hidden'); tbody.classList.add('hidden'); main.setAttribute('style','height: auto !important;'); } else { cfgpage.classList.add('hidden'); tbody.classList.remove('hidden'); main.setAttribute('style',''); }; rebuild = false; }, }), btnhome = make('mujs-btn','github', { title: `GitHub (v${MU.info.script.version.includes('.') ? MU.info.script.version : MU.info.script.version.slice(0,5)})`, innerHTML: ``, onclick: (e) => { e.preventDefault(); openpage('https://github.com/magicoflolis/Userscript-Plus'); } }), btnissue = make('mujs-btn','issue', { title: cfg.lang.issue, innerHTML: ``, onclick: (e) => { e.preventDefault(); openpage('https://github.com/magicoflolis/Userscript-Plus/issues/new'); } }), btngreasy = make('mujs-btn','greasy', { title: 'Greasy Fork', innerHTML: ``, onclick: (e) => { e.preventDefault(); openpage('https://greasyfork.org/scripts/421603'); } }), btnlegacy = make('mujs-btn','legacy', { title: cfg.lang.legacy, innerHTML: ``, onclick: (e) => { e.preventDefault(); openpage('https://cdn.jsdelivr.net/gh/magicoflolis/Userscript-Plus@master/archive/magic-userjs.user.js'); } }); gfcountframe.append(gfcounter); sfcountframe.append(sfcounter); countframe.append(gfcountframe,sfcountframe); fsearch.append(searcher); btnframe.append(fsearch,searchbtn,btncfg,btnissue,btnhome,btngreasy,btnlegacy,closebtn); header.append(countframe,btnframe); main.append(header,tbody,cfgpage); mainframe.append(mainbtn); container.shadowRoot.append(usercss,mainframe,main); for(let b of cfg.blacklist) { if(!b.enabled) continue; const bhref = win.top.document.location.href; if(b.regex) { let reg = new RegExp(b.url,b.flags), testurl = reg.test(bhref); if(!testurl) buildlist(); break; }; if(!bhref.includes(b.url)) buildlist(); break; } makecfg(); } catch(error) {err(error)} }; if(doc.readyState === 'complete') { countsite(); } else { ael(win,'load',countsite); }; }; async function setupConfig() { try { let data = await Promise.all([ checkGMSupport ? MU.getValue('Config',JSON.stringify(defcfg)) : JSON.stringify(defcfg) ]).catch(err); cfg = JSON.parse(localStorage.getItem('MUJSConfig') ?? data[0]); for (const key in defcfg) { if(!Object.prototype.hasOwnProperty.call(cfg, key)) { cfg[key] = defcfg[key]; } else if (key === 'lang') { for (const keyl in defcfg[key]) { if(!Object.prototype.hasOwnProperty.call(cfg[key], keyl)) { cfg[key][keyl] = defcfg[key][keyl]; }; }; } else if (key === 'engines') { for (const key2 in defcfg[key]) { if(!Object.prototype.hasOwnProperty.call(cfg[key], key2)) { cfg[key][key2] = defcfg[key][key2]; }; }; } else if (key === 'blacklist') { for (const key3 in defcfg[key]) { if(!Object.prototype.hasOwnProperty.call(cfg[key], key3)) { cfg[key][key3] = defcfg[key][key3]; }; }; } }; log('Config:',cfg); main(); } catch(error) {err(error)} }; setupConfig(); })();