// ==UserScript==
// @version 6.6.0
// @name Magic Userscript+ : Show Site All UserJS
// @name:ar Magic Userscript+: عرض جميع ملفات UserJS
// @name:es Magic Userscript+: Mostrar sitio todos los UserJS
// @name:fr Magic Userscript+ : Afficher le site Tous les UserJS
// @name:ja Magic Userscript+ : サイトをすべて表示 UserJS
// @name:nl Magic Userscript+: Site alle UserJS tonen
// @name:ru Magic Userscript+: показать сайт всем UserJS
// @name:zh Magic Userscript+ :显示站点所有 UserJS
// @name:zh-CN Magic Userscript+ :显示站点所有 UserJS
// @name:zh-TW Magic Userscript+ :显示站点所有 UserJS
// @description Finds available userscripts for the current webpage.
// @description:ar يبحث عن نصوص المستخدمين المتاحة لصفحة الويب الحالية.
// @description:es Busca los usercripts disponibles para la página web actual.
// @description:fr Recherche les userscripts disponibles pour la page web en cours.
// @description:ja 現在のウェブページで利用可能なユーザスクリプトを検索します。
// @description:nl Zoekt beschikbare gebruikerscripts voor de huidige webpagina.
// @description:ru Находит доступные юзерскрипты для текущей веб-страницы.
// @description:zh 为当前网页查找可用的用户脚本。
// @description:zh-CN 为当前网页查找可用的用户脚本。
// @description:zh-TW 为当前网页查找可用的用户脚本。
// @supportURL https://github.com/magicoflolis/Magic-PH/issues/new/choose
// @namespace https://github.com/magicoflolis/Userscript-Plus
// @homepageURL https://github.com/magicoflolis/Userscript-Plus
// @icon 
// @license MIT
// @compatible chrome
// @compatible firefox
// @compatible edge
// @compatible opera
// @compatible safari
// @connect greasyfork.org
// @connect sleazyfork.org
// @connect github.com
// @connect openuserjs.org
// @grant GM_addElement
// @grant GM_info
// @grant GM_getValue
// @grant GM_openInTab
// @grant GM_setValue
// @grant GM_xmlhttpRequest
// @grant GM.addElement
// @grant GM.info
// @grant GM.getValue
// @grant GM.openInTab
// @grant GM.setValue
// @grant GM.xmlHttpRequest
// @match https://*/*
// @noframes
// @run-at document-start
// @downloadURL none
// ==/UserScript==
(() => {
'use strict';
/******************************************************************************/
/**
* Uncompressed locales + compiler
*
* [_locales](https://github.com/magicoflolis/Userscript-Plus/tree/master/src/_locales)
*
* [languageLoader.js](https://github.com/magicoflolis/Userscript-Plus/blob/master/tools/languageLoader.js)
*/
const translations = {"ar":{"legacy":"يرجى إعادة ضبط التكوين الخاص بك!","createdby":"انشأ من قبل","name":"اسم","daily_installs":"التثبيت اليومي","close":"يغلق","filterA":"منقي","max":"تحقيق أقصى قدر","min":"تصغير","search":"يبحث","search_placeholder":"بحث في البرامج النصية","install":"تثبيت","issue":"إصدار جديد","version_number":"الإصدار","updated":"آخر تحديث","total_installs":"إجمالي التثبيت","ratings":"التقييمات","good":"جيد","ok":"جيد","bad":"سيء","created_date":"تم إنشاؤه","redirect":"شوكة دهنية للكبار","filter":"تصفية اللغات الأخرى","dtime":"عرض المهلة","save":"حفظ","reset":"إعادة تعيين","preview_code":"كود المعاينة","saveFile":"احفظ الملف","newTab":"علامة تبويب جديدة","applies_to":"ينطبق على","license":"الترخيص","no_license":"لا يوجد","antifeatures":"إعلانات","userjs_fullscreen":"ملء الشاشة الكاملة التلقائي","listing_none":"(لا يوجد)","export_config":"تهيئة التصدير","export_theme":"تصدير السمة","import_config":"استيراد تهيئة الاستيراد","import_theme":"استيراد النسق"},"en":{"legacy":"PLEASE RESET YOUR CONFIG!","createdby":"Created by","name":"Name","daily_installs":"Daily Installs","close":"Close","filterA":"Filter","max":"Maximize","min":"Minimize","search":"Search","search_placeholder":"Search for userscripts","install":"Install","issue":"New Issue","version_number":"Version","updated":"Last Updated","total_installs":"Total Installs","ratings":"Ratings","good":"Good","ok":"Ok","bad":"Bad","created_date":"Created","redirect":"Greasy Fork for adults","filter":"Filter out other languages","dtime":"Display Timeout","save":"Save","reset":"Reset","preview_code":"Preview Code","saveFile":"Save File","newTab":"New Tab","applies_to":"Applies to","license":"License","no_license":"N/A","antifeatures":"Antifeatures","userjs_fullscreen":"Automatic Fullscreen","listing_none":"(None)","export_config":"Export Config","export_theme":"Export Theme","import_config":"Import Config","import_theme":"Import Theme"},"en_GB":{"legacy":"PLEASE RESET YOUR CONFIG!","createdby":"Created by","name":"Name","daily_installs":"Daily Installs","close":"Close","filterA":"Filter","max":"Maximize","min":"Minimize","search":"Search","search_placeholder":"Search scripts","install":"Install","issue":"New Issue","version_number":"Version","updated":"Last Updated","total_installs":"Total Installs","ratings":"Ratings","good":"Good","ok":"Ok","bad":"Bad","created_date":"Created","redirect":"Greasy Fork for adults","filter":"Filter out other languages","dtime":"Display Timeout","save":"Save","reset":"Reset","preview_code":"Preview Code","saveFile":"Save File","newTab":"New Tab","applies_to":"Applies to","license":"License","no_license":"N/A","antifeatures":"Antifeatures","userjs_fullscreen":"Automatic Fullscreen","listing_none":"(None)","export_config":"Export Config","export_theme":"Export Theme","import_config":"Import Config","import_theme":"Import Theme"},"es":{"legacy":"¡POR FAVOR RESTABLECE TU CONFIGURACIÓN!","createdby":"Creado por","name":"Nombre","daily_installs":"Instalaciones diarias","close":"Ya no se muestra","filterA":"Filtro","max":"Maximizar","min":"Minimizar","search":"Busque en","search_placeholder":"Buscar userscripts","install":"Instalar","issue":"Nueva edición","version_number":"Versión","updated":"Última actualización","total_installs":"Total de instalaciones","ratings":"Clasificaciones","good":"Bueno","ok":"Ok","bad":"Malo","created_date":"Creado","redirect":"Greasy Fork para adultos","filter":"Filtrar otros idiomas","dtime":"Mostrar el tiempo de espera","save":"Guardar","reset":"Reiniciar","preview_code":"Vista previa del código","saveFile":"Guardar archivo","newTab":"Guardar archivo","applies_to":"Se aplica a","license":"Licencia","no_license":"Desconocida","antifeatures":"Características indeseables","userjs_fullscreen":"Pantalla completa automática","listing_none":"(Ninguno)","export_config":"Exportar configuración","export_theme":"Exportar tema","import_config":"Importar configuración","import_theme":"Importar tema"},"fr":{"legacy":"VEUILLEZ RÉINITIALISER VOTRE CONFIG !","createdby":"Créé par","name":"Nom","daily_installs":"Installations quotidiennes","close":"Ne plus montrer","filterA":"Filtre","max":"Maximiser","min":"Minimiser","search":"Recherche","search_placeholder":"Rechercher des userscripts","install":"Installer","issue":"Nouveau numéro","version_number":"Version","updated":"Dernière mise à jour","total_installs":"Total des installations","ratings":"Notations","good":"Bon","ok":"Ok","bad":"Mauvais","created_date":"Créé","redirect":"Greasy Fork pour les adultes","filter":"Filtrer les autres langues","dtime":"Délai d'affichage","save":"Sauvez","reset":"Réinitialiser","preview_code":"Prévisualiser le code","saveFile":"Enregistrer le fichier","newTab":"Nouvel onglet","applies_to":"S'applique à","license":"Licence","no_license":"N/A","antifeatures":"Antifeatures","userjs_fullscreen":"Plein écran automatique","listing_none":"(Aucun)","export_config":"Export Config","export_theme":"Exporter le thème","import_config":"Importer la configuration","import_theme":"Importer le thème"},"ja":{"legacy":"設定をリセットしてください。","createdby":"によって作成された","name":"名前","daily_installs":"デイリーインストール","close":"表示されなくなりました","filterA":"フィルター","max":"最大化","min":"ミニマム","search":"検索","search_placeholder":"ユーザースクリプトの検索","install":"インストール","issue":"新刊のご案内","version_number":"バージョン","updated":"最終更新日","total_installs":"総インストール数","ratings":"レーティング","good":"グッド","ok":"良い","bad":"悪い","created_date":"作成","redirect":"大人のGreasyfork","filter":"他の言語をフィルタリングする","dtime":"表示タイムアウト","save":"拯救","reset":"リセット","preview_code":"コードのプレビュー","saveFile":"ファイルを保存","newTab":"新しいタブ","applies_to":"適用対象","license":"ライセンス","no_license":"不明","antifeatures":"アンチ機能","userjs_fullscreen":"自動フルスクリーン","listing_none":"(なし)","export_config":"エクスポート設定","export_theme":"テーマのエクスポート","import_config":"設定のインポート","import_theme":"テーマのインポート"},"nl":{"legacy":"RESET UW CONFIG!","createdby":"Gemaakt door","name":"Naam","daily_installs":"Dagelijkse Installaties","close":"Sluit","filterA":"Filter","max":"Maximaliseer","min":"Minimaliseer","search":"Zoek","search_placeholder":"Zoeken naar gebruikersscripts","install":"Installeer","issue":"Nieuw Issue","version_number":"Versie","updated":"Laatste Update","total_installs":"Totale Installaties","ratings":"Beoordeling","good":"Goed","ok":"Ok","bad":"Slecht","created_date":"Aangemaakt","redirect":"Greasy Fork voor volwassenen","filter":"Filter andere talen","dtime":"Weergave timeout","save":"Opslaan","reset":"Opnieuw instellen","preview_code":"Voorbeeldcode","saveFile":"Bestand opslaan","newTab":"Nieuw tabblad","applies_to":"Geldt voor","license":"Licentie","no_license":"N.v.t.","antifeatures":"Functies voor eigen gewin","userjs_fullscreen":"Automatisch volledig scherm","listing_none":"(Geen)","export_config":"Configuratie exporteren","export_theme":"Thema exporteren","import_config":"Configuratie importeren","import_theme":"Thema importeren"},"ru":{"legacy":"ПОЖАЛУЙСТА, СБРОСЬТЕ КОНФИГ!","createdby":"Сделано","name":"Имя","daily_installs":"Ежедневные установки","close":"Больше не показывать","filterA":"Фильтр","max":"Максимизировать","min":"Минимизировать","search":"Поиск","search_placeholder":"Поиск юзерскриптов","install":"Установите","issue":"Новый выпуск","version_number":"Версия","updated":"Последнее обновление","total_installs":"Всего установок","ratings":"Рейтинги","good":"Хорошо","ok":"Хорошо","bad":"Плохо","created_date":"Создано","redirect":"Greasy Fork для взрослых","filter":"Отфильтровать другие языки","dtime":"Тайм-аут отображения","save":"Сохранить","reset":"Перезагрузить","preview_code":"Предварительный просмотр кода","saveFile":"Сохранить файл","newTab":"Новая вкладка","applies_to":"Применяется к","license":"Лицензия","no_license":"Недоступно","antifeatures":"Нежелательная функциональность","userjs_fullscreen":"Автоматический полноэкранный режим","listing_none":"(нет)","export_config":"Экспорт конфигурации","export_theme":"Экспорт темы","import_config":"Импорт конфигурации","import_theme":"Импортировать тему"},"zh":{"legacy":"请重置您的配置!","createdby":"由...制作","name":"姓名","daily_installs":"日常安装","close":"不再显示","filterA":"过滤器","max":"最大化","min":"最小化","search":"搜索","search_placeholder":"搜索用户脚本","install":"安装","issue":"新问题","version_number":"版本","updated":"最后更新","total_installs":"总安装量","ratings":"评级","good":"好的","ok":"好的","bad":"不好","created_date":"创建","redirect":"大人的Greasyfork","filter":"过滤掉其他语言","dtime":"显示超时","save":"拯救","reset":"重置","preview_code":"预览代码","saveFile":"保存存档","newTab":"新标签","applies_to":"适用于","license":"许可证","no_license":"暂无","antifeatures":"可能不受欢迎的功能","userjs_fullscreen":"自动全屏","listing_none":"(无)","export_config":"导出配置","export_theme":"导出主题","import_config":"导入配置","import_theme":"导入主题"},"zh_CN":{"legacy":"请重置您的配置!","createdby":"由...制作","name":"姓名","daily_installs":"日常安装","close":"不再显示","filterA":"过滤器","max":"最大化","min":"最小化","search":"搜索","search_placeholder":"搜索用户脚本","install":"安装","issue":"新问题","version_number":"版本","updated":"最后更新","total_installs":"总安装量","ratings":"评级","good":"好的","ok":"好的","bad":"不好","created_date":"创建","redirect":"大人的Greasyfork","filter":"过滤掉其他语言","dtime":"显示超时","save":"拯救","reset":"重置","preview_code":"预览代码","saveFile":"保存存档","newTab":"新标签","applies_to":"适用于","license":"许可证","no_license":"暂无","antifeatures":"可能不受欢迎的功能","userjs_fullscreen":"自动全屏","listing_none":"(无)","export_config":"导出配置","export_theme":"导出主题","import_config":"导入配置","import_theme":"导入主题"},"zh_TW":{"legacy":"请重置您的配置!","createdby":"由...制作","name":"姓名","daily_installs":"日常安装","close":"不再显示","filterA":"过滤器","max":"最大化","min":"最小化","search":"搜索","search_placeholder":"搜索用户脚本","install":"安装","issue":"新问题","version_number":"版本","updated":"最后更新","total_installs":"总安装量","ratings":"评级","good":"好的","ok":"好的","bad":"不好","created_date":"创建","redirect":"大人的Greasyfork","filter":"过滤掉其他语言","dtime":"显示超时","save":"拯救","reset":"重置","preview_code":"预览代码","saveFile":"保存存档","newTab":"新标签","applies_to":"适用于","license":"许可证","no_license":"暂无","antifeatures":"可能不受欢迎的功能","userjs_fullscreen":"自动全屏","listing_none":"(无)","export_config":"导出配置","export_theme":"导出主题","import_config":"导入配置","import_theme":"导入主题"}};
/**
* To compile this CSS `pnpm run build:Sass`
*
* [Uncompiled Cascading Style Sheet](https://github.com/magicoflolis/Userscript-Plus/tree/master/src/sass)
*/
const main_css = `mujs-root{--mujs-even-row: hsl(222, 14%, 22%);--mujs-odd-row: hsl(222, 14%, 11%);--mujs-even-err: hsl(0, 100%, 22%);--mujs-odd-err: hsl(0, 100%, 11%);--mujs-background-color: hsl(222, 14%, 33%);--mujs-gf-color: hsl(204, 100%, 40%);--mujs-sf-color: hsl(12, 86%, 50%);--mujs-border-b-color: hsla(0, 0%, 0%, 0);--mujs-gf-btn-color: hsl(211, 87%, 56%);--mujs-sf-btn-color: hsl(12, 86%, 50%);--mujs-sf-txt-color: hsl(12, 79%, 55%);--mujs-txt-color: hsl(0, 0%, 100%);--mujs-chck-color: hsla(0, 0%, 100%, 0.568);--mujs-chck-gf: hsla(197, 100%, 50%, 0.568);--mujs-chck-git: hsla(213, 13%, 16%, 0.568);--mujs-chck-open: hsla(12, 86%, 50%, 0.568);--mujs-placeholder: hsl(81, 56%, 54%);--mujs-position-top: unset;--mujs-position-bottom: 1em;--mujs-position-left: unset;--mujs-position-right: 1em;--mujs-font-family: Arial, Helvetica, sans-serif;font-family:var(--mujs-font-family, Arial, Helvetica, sans-serif);text-rendering:optimizeLegibility;word-break:normal;font-size:14px;color:var(--mujs-txt-color, hsl(0, 0%, 100%))}mujs-root *{scrollbar-color:var(--mujs-txt-color, hsl(0, 0%, 100%)) #2e323d;scrollbar-width:thin}@supports not (scrollbar-width: thin){mujs-root * ::-webkit-scrollbar{width:1.4vw;height:3.3vh}mujs-root * ::-webkit-scrollbar-track{background-color:#2e323d;border-radius:16px;margin-top:3px;margin-bottom:3px;box-shadow:inset 0 0 6px rgba(0,0,0,.3)}mujs-root * ::-webkit-scrollbar-thumb{border-radius:16px;background-color:var(--mujs-txt-color, hsl(0, 0%, 100%));background-image:-webkit-linear-gradient(45deg, hsla(0, 0%, 100%, 0.2) 25%, transparent 25%, transparent 50%, hsla(0, 0%, 100%, 0.2) 50%, hsla(0, 0%, 100%, 0.2) 75%, transparent 75%, transparent)}mujs-root * ::-webkit-scrollbar-thumb:hover{background:var(--mujs-txt-color, hsl(0, 0%, 100%))}}mu-js{line-height:normal}mujs-section>label,.mujs-homepag e,td.mujs-list,.install{font-size:16px}.install,.mujs-homepage{font-weight:700}mujs-section>label,td.mujs-list{font-weight:500}.mujs-sty-flex>mujs-btn{margin:auto}.mujs-invalid{border-radius:8px !important;border-width:2px !important;border-style:solid !important;border-color:red !important}mujs-tabs,mujs-column,mujs-row,.mujs-sty-flex{display:flex}mujs-column,mujs-row{gap:.5em}@media screen and (max-width: 800px){mujs-column{flex-flow:row wrap}}mujs-column count-frame[data-counter=greasyfork]{background:var(--mujs-gf-color, hsl(204, 100%, 40%))}mujs-column count-frame[data-counter=sleazyfork]{background:var(--mujs-sf-color, hsl(12, 86%, 50%))}mujs-column count-frame[data-counter=github]{background:#23282e}mujs-column count-frame[data-counter=openuserjs]{background:rgba(237,62,18,.568)}mujs-row{flex-flow:column wrap}mu-js{cursor:default}.hidden{display:none !important;z-index:-1 !important}mujs-main{width:100%;width:-moz-available;width:-webkit-fill-available;background:var(--mujs-background-color, hsl(222, 14%, 33%)) !important;border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));border-radius:16px}@media screen and (max-height: 720px){mujs-main:not(.webext-page){height:100% !important;bottom:0rem !important;right:0rem !important;margin:0rem !important}}mujs-main.expanded{height:100% !important;bottom:0rem !important}mujs-main:not(.webext-page){position:fixed;height:492px}mujs-main:not(.webext-page):not(.expanded){margin-left:1rem;margin-right:1rem;right:1rem;bottom:1rem}mujs-main:not(.hidden){z-index:100000000000000000 !important;display:flex !important;flex-direction:column !important}mujs-main>*{width:100%;width:-moz-available;width:-webkit-fill-available}mujs-main mujs-toolbar{order:0;padding:.5em;display:flex;place-content:space-between}mujs-main mujs-toolbar mujs-tabs{overflow:hidden;order:0}mujs-main mujs-toolbar mujs-column{flex-flow:row nowrap;order:999999999999}mujs-main mujs-toolbar>*{width:-webkit-fit-content;width:-moz-fit-content;width:fit-content}mujs-main mujs-tabs{gap:.5em;text-align:center;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;flex-flow:row wrap}mujs-main mujs-tabs mujs-tab{padding:.25em;min-width:150px;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;height:-webkit-fit-content;height:-moz-fit-content;height:fit-content;display:flex;place-content:space-between;border:1px solid rgba(0,0,0,0);border-radius:4px;background:rgba(0,0,0,0)}mujs-main mujs-tabs mujs-tab.active{background:var(--mujs-even-row, hsl(222, 14%, 18%))}mujs-main mujs-tabs mujs-tab:not(.active):hover{background:var(--mujs-even-row, hsl(222, 14%, 18%))}mujs-main mujs-tabs mujs-tab mujs-host{float:left;overflow:auto;overflow-wrap:break-word;text-overflow:ellipsis;white-space:nowrap}mujs-main mujs-tabs mujs-tab mu-js{float:right}mujs-main mujs-tabs mujs-addtab{order:999999999999;font-size:20px;padding:0px .25em}mujs-main mujs-tabs mujs-addtab:hover{background:var(--mujs-even-row, hsl(222, 14%, 18%))}mujs-main mujs-tab,mujs-main mujs-btn,mujs-main input{width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;height:-webkit-fit-content;height:-moz-fit-content;height:fit-content}mujs-main input{background:rgba(0,0,0,0);color:var(--mujs-txt-color, hsl(0, 0%, 100%))}mujs-main input:not([type=checkbox]){border:rgba(0,0,0,0);outline:none !important}mujs-main textarea{background:inherit;overflow-y:auto;color:var(--mujs-placeholder, hsl(81, 56%, 54%));border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));border-radius:10px;resize:vertical;outline:none;font-family:monospace;font-size:14px}mujs-main textarea:focus{outline:none}mujs-main th,mujs-main .mujs-cfg *:not(input[type=password],input[type=text],input[type=number]){-webkit-user-select:none !important;-moz-user-select:none !important;-ms-user-select:none !important;user-select:none !important}mujs-main .mujs-footer{order:3;overflow-x:hidden;text-align:center;border-radius:16px}mujs-main .mujs-footer>*{min-height:50px}mujs-main .mujs-footer .error:nth-child(even){background:var(--mujs-even-err, hsl(0, 100%, 22%)) !important}mujs-main .mujs-footer .error:nth-child(odd){background:var(--mujs-odd-err, hsl(0, 100%, 11%)) !important}.mainframe{background:rgba(0,0,0,0);position:fixed;bottom:var(--mujs-position-bottom, 1rem);right:var(--mujs-position-right, 1rem);top:var(--mujs-position-top, unset);left:var(--mujs-position-left, unset)}.mainframe count-frame{width:fit-content;width:-moz-fit-content;width:-webkit-fit-content;height:auto;padding:14px 16px}.mainframe.error{opacity:1 !important}.mainframe.error count-frame{background:var(--mujs-even-err, hsl(0, 100%, 22%)) !important}.mainframe:not(.hidden){z-index:100000000000000000 !important;display:block}count-frame{border-radius:1000px;margin:0px 3px;padding:4px 6px;border:2px solid var(--mujs-border-b-color, hsla(0, 0%, 0%, 0));font-size:16px;font-weight:400;display:inline-block;text-align:center;min-width:1em;background:var(--mujs-background-color, hsl(222, 14%, 33%));-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}mujs-header{order:1;display:flex;border-bottom:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));padding-left:.5em;padding-right:.5em;padding-bottom:.5em;font-size:1em;place-content:space-between;height:fit-content;height:-moz-fit-content;height:-webkit-fit-content;gap:1em}mujs-header>*:not(mujs-url){height:fit-content;height:-moz-fit-content;height:-webkit-fit-content}mujs-header mujs-url{order:0;flex-grow:1}mujs-header mujs-url>input{width:100%;height:100%;background:var(--mujs-even-row, hsl(222, 14%, 18%));border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));border-radius:4px}mujs-header .rate-container{order:1}mujs-header .mujs-sty-flex{order:2}mujs-header .btn-frame{order:999999999999}mujs-body{order:2;overflow-x:hidden;padding:0px;height:100%;border:1px solid var(--mujs-border-b-color, hsla(0, 0%, 0%, 0));border-bottom-left-radius:16px;border-bottom-right-radius:16px}mujs-body .mujs-ratings{padding:0 .25em;border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));border-radius:1000px;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content}mujs-body mu-jsbtn{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}mujs-body table,mujs-body th,mujs-body td{border-collapse:collapse}mujs-body table{width:100%;width:-moz-available;width:-webkit-fill-available}@media screen and (max-width: 1180px){mujs-body table thead>tr{display:table-column}mujs-body table .frame:not(.webext-page){width:100%;display:flex;flex-flow:row wrap;align-items:center;border-bottom:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));padding-top:.5em;padding-bottom:.5em}mujs-body table .frame:not(.webext-page) td{margin:auto;border-bottom:1px solid rgba(0,0,0,0)}mujs-body table .frame:not(.webext-page) td>mujs-a,mujs-body table .frame:not(.webext-page) td>mu-js,mujs-body table .frame:not(.webext-page) td>mujs-column{text-align:center;justify-content:center}mujs-body table .frame:not(.webext-page) td:not(.mujs-name,.install-btn){width:25%}mujs-body table .frame:not(.webext-page) .mujs-name{width:100%}}mujs-body table th,mujs-body table td{border-bottom:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%))}mujs-body table th{position:-webkit-sticky;position:sticky;top:0;background:rgba(72,79,96,.75)}mujs-body table th.mujs-header-name{width:50%}@media screen and (max-width: 800px){mujs-body table th.mujs-header-name{width:auto !important}}mujs-body table .frame:nth-child(even){background:var(--mujs-even-row, hsl(222, 14%, 18%)) !important}mujs-body table .frame:nth-child(odd){background:var(--mujs-odd-row, hsl(222, 14%, 33%)) !important}mujs-body table .frame:not([data-engine=sleazyfork],[data-engine=greasyfork]) mujs-a{color:var(--mujs-sf-txt-color, hsl(12, 79%, 55%))}mujs-body table .frame:not([data-engine=sleazyfork],[data-engine=greasyfork]) mu-jsbtn{background:var(--mujs-sf-btn-color, hsl(12, 86%, 50%));border-color:var(--mujs-sf-btn-color, hsl(12, 86%, 50%))}mujs-body table .frame:not([data-engine=sleazyfork],[data-engine=greasyfork]) mu-jsbtn:hover{background:var(--mujs-sf-txt-color, hsl(12, 79%, 55%));border-color:var(--mujs-sf-txt-color, hsl(12, 79%, 55%))}mujs-body table .frame[data-engine=sleazyfork] mujs-a,mujs-body table .frame[data-engine=greasyfork] mujs-a{color:var(--mujs-gf-color, hsl(197, 100%, 50%))}mujs-body table .frame[data-engine=sleazyfork] mujs-a:hover,mujs-body table .frame[data-engine=greasyfork] mujs-a:hover{color:var(--mujs-gf-btn-color, hsl(211, 87%, 56%))}mujs-body table .frame[data-engine=sleazyfork] mu-jsbtn,mujs-body table .frame[data-engine=greasyfork] mu-jsbtn{background:var(--mujs-gf-color, hsl(204, 100%, 40%));border-color:var(--mujs-gf-color, hsl(204, 100%, 40%))}mujs-body table .frame[data-engine=sleazyfork] mu-jsbtn:hover,mujs-body table .frame[data-engine=greasyfork] mu-jsbtn:hover{background:var(--mujs-gf-btn-color, hsl(211, 87%, 56%));border-color:var(--mujs-gf-btn-color, hsl(211, 87%, 56%))}mujs-body table .frame[data-good] mujs-a,mujs-body table .frame[data-author] mujs-a{color:var(--mujs-placeholder, hsl(81, 56%, 54%))}mujs-body table .frame[data-good] mujs-a:hover,mujs-body table .frame[data-author] mujs-a:hover{color:#80ab30}mujs-body table .frame[data-good] .mujs-list,mujs-body table .frame[data-author] .mujs-list{color:#fff}mujs-body table .frame[data-good] mu-jsbtn,mujs-body table .frame[data-author] mu-jsbtn{color:#20385a;background:var(--mujs-placeholder, hsl(81, 56%, 54%));border-color:var(--mujs-placeholder, hsl(81, 56%, 54%))}mujs-body table .frame[data-good] mu-jsbtn:hover,mujs-body table .frame[data-author] mu-jsbtn:hover{background:#b5d874;border-color:#b5d874}mujs-body table .frame .mujs-ratings[data-el=good]{border-color:#393;background-color:rgba(51,153,51,.102);color:#3f3}mujs-body table .frame .mujs-ratings[data-el=ok]{border-color:#990;background-color:rgba(153,153,0,.102);color:#ff0}mujs-body table .frame .mujs-ratings[data-el=bad]{border-color:#900;background-color:rgba(153,51,51,.102);color:red}mujs-body table .frame svg{fill:currentColor;width:14px;height:14px;background:rgba(0,0,0,0)}mujs-body table .frame>td:not(.mujs-name){text-align:center}mujs-body table .frame>.mujs-name>mujs-a{width:-webkit-fit-content;width:-moz-fit-content;width:fit-content}mujs-body table .frame>.mujs-name mu-jsbtn,mujs-body table .frame>.mujs-name mu-js{height:-webkit-fit-content;height:-moz-fit-content;height:fit-content}mujs-body table .frame>.mujs-name>mu-jsbtn{margin:auto}mujs-body table .frame>.mujs-name>mujs-column>mu-jsbtn{padding:0px 7px}mujs-body table .frame>.mujs-uframe>mujs-a{font-size:16px;font-weight:500;padding-left:.5rem;padding-right:.5rem}mujs-body table .frame [data-el=more-info]>mujs-row{gap:.25em}mujs-body table .frame [data-el=matches]{gap:.25em;max-width:40em}mujs-body table .frame [data-el=matches] .mujs-grants{display:inline-flex;flex-flow:row wrap;overflow:auto;overflow-wrap:break-word;text-overflow:ellipsis;white-space:nowrap;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;max-height:5em;gap:.2em}mujs-body table .frame [data-el=matches] .mujs-grants>mujs-a{display:inline}mujs-body table .frame [data-el=matches] .mujs-grants>mujs-a:not([data-command]){cursor:default !important;color:var(--mujs-txt-color, hsl(0, 0%, 100%))}mujs-body table .frame [data-el=matches] .mujs-grants>mujs-a::after{content:", ";color:var(--mujs-txt-color, hsl(0, 0%, 100%))}mujs-body table .frame [data-el=matches] .mujs-grants>mujs-a:last-child::after{content:""}@media screen and (max-width: 1150px){.mujs-cfg{margin:0px auto 1rem auto !important}}.mujs-cfg{height:fit-content;height:-moz-fit-content;height:-webkit-fit-content}@media screen and (max-height: 720px){.mujs-cfg:not(.webext-page){height:100%;height:-moz-available;height:-webkit-fill-available;width:100%;width:-moz-available;width:-webkit-fill-available;overflow-x:auto;padding:.5em}}.mujs-cfg mujs-section{border-radius:16px;padding:.5em}.mujs-cfg mujs-section:nth-child(even){background:var(--mujs-even-row, hsl(222, 14%, 18%)) !important}.mujs-cfg mujs-section:nth-child(odd){background:var(--mujs-odd-row, hsl(222, 14%, 33%)) !important}.mujs-cfg mujs-section>label{display:flex;justify-content:space-between}.mujs-cfg mujs-section>label input:not([type=checkbox]){font-size:14px;position:relative;border-radius:4px;border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%))}.mujs-cfg mujs-section>label input[type=text]::-webkit-input-placeholder{color:var(--mujs-placeholder, hsl(81, 56%, 54%))}.mujs-cfg mujs-section>label input[type=text]::-moz-placeholder{color:var(--mujs-placeholder, hsl(81, 56%, 54%))}.mujs-cfg mujs-section>label input[type=text]:-ms-input-placeholder{color:var(--mujs-placeholder, hsl(81, 56%, 54%))}.mujs-cfg mujs-section>label input[type=text]::-ms-input-placeholder{color:var(--mujs-placeholder, hsl(81, 56%, 54%))}.mujs-cfg mujs-section>label input[type=text]::placeholder{color:var(--mujs-placeholder, hsl(81, 56%, 54%))}.mujs-cfg .mujs-inlab{position:relative;width:38px}.mujs-cfg .mujs-inlab input[type=checkbox]{display:none}.mujs-cfg .mujs-inlab input[type=checkbox]:checked+label{margin-left:0;background:var(--mujs-chck-color, hsla(0, 0%, 100%, 0.568))}.mujs-cfg .mujs-inlab input[type=checkbox]:checked+label:before{right:0px}.mujs-cfg .mujs-inlab input[type=checkbox][data-name=greasyfork]:checked+label,.mujs-cfg .mujs-inlab input[type=checkbox][data-name=sleazyfork]:checked+label{background:var(--mujs-chck-gf, hsla(197, 100%, 50%, 0.568))}.mujs-cfg .mujs-inlab input[type=checkbox][data-name=openuserjs]:checked+label{background:var(--mujs-chck-open, hsla(12, 86%, 50%, 0.568))}.mujs-cfg .mujs-inlab input[type=checkbox][data-name=github]:checked+label{background:var(--mujs-chck-git, hsla(213, 13%, 16%, 0.568))}.mujs-cfg .mujs-inlab label{padding:0;display:block;overflow:hidden;height:16px;border-radius:20px;border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%))}.mujs-cfg .mujs-inlab label:before{content:"";display:block;width:20px;height:20px;margin:-2px;background:var(--mujs-txt-color, hsl(0, 0%, 100%));position:absolute;top:0;right:20px;border-radius:20px}.mujs-cfg .mujs-sty-flex mujs-btn[data-command=reset]{background:var(--mujs-sf-btn-color, hsl(12, 86%, 50%));border-color:var(--mujs-sf-btn-color, hsl(12, 86%, 50%))}.mujs-cfg .mujs-sty-flex mujs-btn[data-command=reset]:hover{background:var(--mujs-sf-txt-color, hsl(12, 79%, 55%));border-color:var(--mujs-sf-txt-color, hsl(12, 79%, 55%))}.mujs-cfg .mujs-sty-flex mujs-btn[data-command=save]{background:var(--mujs-gf-color, hsl(204, 100%, 40%));border-color:var(--mujs-gf-color, hsl(204, 100%, 40%))}.mujs-cfg .mujs-sty-flex mujs-btn[data-command=save]:hover{background:var(--mujs-gf-btn-color, hsl(211, 87%, 56%));border-color:var(--mujs-gf-btn-color, hsl(211, 87%, 56%))}.mujs-cfg:not(.webext-page){margin:1rem 25rem}mujs-a{display:inline-block}.mujs-name{display:flex;flex-flow:column wrap;gap:.5em}.mujs-name span{font-size:.8em !important}mujs-btn{font-style:normal;font-weight:500;font-variant:normal;text-transform:none;text-rendering:auto;text-align:center;border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));font-size:16px;border-radius:4px;line-height:1;padding:6px 15px}mujs-btn svg{fill:var(--mujs-txt-color, hsl(0, 0%, 100%));width:14px;height:14px}mu-jsbtn{font-size:14px;border-radius:4px;font-style:normal;padding:7px 15%;font-weight:400;font-variant:normal;line-height:normal;display:block;text-align:center}mujs-a,mu-jsbtn,.mujs-pointer,.mujs-cfg mujs-section *:not(input[type=password],input[type=text],input[type=number]),.mainbtn,.mainframe,mujs-btn{cursor:pointer !important}
`;
let userjs = self.userjs;
/**
* Skip text/plain documents, based on uBlock Origin `vapi.js` file
*
* [Source Code](https://github.com/gorhill/uBlock/blob/master/platform/common/vapi.js)
*/
if (
(document instanceof Document ||
(document instanceof XMLDocument && document.createElement('div') instanceof HTMLDivElement)) &&
/^image\/|^text\/plain/.test(document.contentType || '') === false &&
(self.userjs instanceof Object === false || userjs.UserJS !== true)
) {
userjs = self.userjs = { UserJS: true };
}
if (!(typeof userjs === 'object' && userjs.UserJS)) {
return;
}
const inIframe = () => {
try {
return window.self !== window.top;
} catch (e) {
return true;
}
}
if (inIframe()) {
return;
}
/******************************************************************************/
// #region Console
const dbg = (...msg) => {
const dt = new Date();
console.debug(
'[%cMagic Userscript+%c] %cDBG',
'color: rgb(29, 155, 240);',
'',
'color: rgb(255, 212, 0);',
`[${dt.getHours()}:${('0' + dt.getMinutes()).slice(-2)}:${('0' + dt.getSeconds()).slice(-2)}]`,
...msg
);
};
const err = (...msg) => {
console.error(
'[%cMagic Userscript+%c] %cERROR',
'color: rgb(29, 155, 240);',
'',
'color: rgb(249, 24, 128);',
...msg
);
for (const ex of msg) {
if (typeof ex === 'object' && 'cause' in ex && typeof alert !== 'undefined') {
alert(`[Magic Userscript+] (${ex.cause}) ${ex.message}`);
}
}
};
const info = (...msg) => {
console.info(
'[%cMagic Userscript+%c] %cINF',
'color: rgb(29, 155, 240);',
'',
'color: rgb(0, 186, 124);',
...msg
);
};
const log = (...msg) => {
console.log(
'[%cMagic Userscript+%c] %cLOG',
'color: rgb(29, 155, 240);',
'',
'color: rgb(219, 160, 73);',
...msg
);
};
// #endregion
let cfg = {};
let ghMsg = null;
const winLocation = window.top.location;
/** @type { import("../typings/UserJS.d.ts").safeSelf } */
function safeSelf() {
if (userjs.safeSelf) {
return userjs.safeSelf;
}
const g = globalThis;
const safe = {
XMLHttpRequest: g.XMLHttpRequest,
createElement: g.document.createElement.bind(g.document),
createElementNS: g.document.createElementNS.bind(g.document),
createTextNode: g.document.createTextNode.bind(g.document),
setTimeout: g.setTimeout,
clearTimeout: g.clearTimeout
};
userjs.safeSelf = safe;
return safe;
}
// Lets highlight me :)
const authorID = 166061;
// Some UserJS I personally enjoy
const goodUserJS = [
423001,
376510,
23840,
40525,
6456,
'https://github.com/TagoDR/MangaOnlineViewer/raw/master/Manga_OnlineViewer.user.js',
'https://github.com/jesus2099/konami-command/raw/master/INSTALL-USER-SCRIPT.user.js',
'https://github.com/TagoDR/MangaOnlineViewer/raw/master/dist/Manga_OnlineViewer_Adult.user.js'
];
// Unsupport webpages for each engine
const engineUnsupported = {
greasyfork: ['pornhub.com'],
sleazyfork: ['pornhub.com'],
openuserjs: [],
github: []
};
const getUAData = () => {
if (typeof userjs.isMobile !== 'undefined') {
return userjs.isMobile;
}
try {
const { platform, mobile } = navigator.userAgentData ?? {};
userjs.isMobile =
/Mobile|Tablet/.test(navigator.userAgent ?? '') ||
mobile ||
/Android|Apple/.test(platform ?? '');
} catch (ex) {
err({ cause: 'getUAData', message: ex.message });
userjs.isMobile = false;
}
return userjs.isMobile;
};
const isMobile = getUAData();
const isGM = typeof GM !== 'undefined';
const MU = {};
const META_START_COMMENT = '// ==UserScript==';
const META_END_COMMENT = '// ==/UserScript==';
const TLD_EXPANSION = ['com', 'net', 'org', 'de', 'co.uk'];
const APPLIES_TO_ALL_PATTERNS = [
'http://*',
'https://*',
'http://*/*',
'https://*/*',
'http*://*',
'http*://*/*',
'*',
'*://*',
'*://*/*',
'http*'
];
const builtinList = {
local: /localhost|router|gov|(\d+\.){3}\d+/,
finance:
/school|pay|bank|money|cart|checkout|authorize|bill|wallet|venmo|zalo|skrill|bluesnap|coin|crypto|currancy|insurance|finance/,
social: /login|join|signin|signup|sign-up|password|reset|password_reset/,
unsupported: {
host: 'fakku.net',
pathname: '/hentai/.+/read/page/.+'
}
};
const defcfg = {
cache: true,
codePreview: false,
autoexpand: false,
filterlang: false,
sleazyredirect: false,
time: 10000,
blacklist: ['userjs-local', 'userjs-finance', 'userjs-social', 'userjs-unsupported'],
engines: [
{
enabled: true,
name: 'greasyfork',
url: 'https://greasyfork.org'
},
{
enabled: false,
name: 'sleazyfork',
url: 'https://sleazyfork.org'
},
{
enabled: false,
name: 'openuserjs',
url: 'https://openuserjs.org/?q='
},
{
enabled: false,
name: 'github',
url: 'https://api.github.com/search/code?q=',
token: ''
}
],
theme: {
'even-row': '',
'odd-row': '',
'even-err': '',
'odd-err': '',
'background-color': '',
'gf-color': '',
'sf-color': '',
'border-b-color': '',
'gf-btn-color': '',
'sf-btn-color': '',
'sf-txt-color': '',
'txt-color': '',
'chck-color': '',
'chck-gf': '',
'chck-git': '',
'chck-open': '',
placeholder: '',
'position-top': '',
'position-bottom': '',
'position-left': '',
'position-right': '',
'font-family': ''
},
recommend: {
author: true,
others: true
}
};
class LanguageHandler {
constructor() {
this.current = (navigator.language ?? 'en').split('-')[0] ?? 'en';
this.cache = [];
const languages = navigator.languages ?? [];
for (const nlang of languages) {
const lg = nlang.split('-')[0];
if (this.cache.indexOf(lg) === -1) {
this.cache.push(lg);
}
}
if (!this.cache.includes(this.current)) {
this.cache.push(this.current);
}
}
}
const language = new LanguageHandler();
const i18n$ = (...args) => {
const list = translations[cfg.language] ?? translations[language.current];
const arr = [];
for (const arg of args) {
arr.push(list[arg]);
}
return arr.length !== 1 ? arr : arr[0];
};
// #region Utilities
/**
* @type { import("../typings/UserJS.d.ts").hasOwn }
*/
const hasOwn = (...args) => {
if (typeof Object.hasOwn !== 'undefined') {
return Object.hasOwn(...args);
}
return Object.prototype.hasOwnProperty.call(...args);
};
/**
* @type { import("../typings/UserJS.d.ts").objToStr }
*/
const objToStr = (obj) => {
return Object.prototype.toString.call(obj);
};
/**
* @type { import("../typings/UserJS.d.ts").strToURL }
*/
const strToURL = (str) => {
try {
return objToStr(str).includes('URL') ? str : new URL(str || window.location || 'about:blank');
} catch (ex) {
err({ cause: 'strToURL', message: ex.message });
return window.location;
}
};
/**
* @type { import("../typings/UserJS.d.ts").isRegExp }
*/
const isRegExp = (obj) => {
const s = objToStr(obj);
return s.includes('RegExp');
};
/**
* @type { import("../typings/UserJS.d.ts").isElem }
*/
const isElem = (obj) => {
const s = objToStr(obj);
return s.includes('Element');
};
/**
* @type { import("../typings/UserJS.d.ts").isObj }
*/
const isObj = (obj) => {
const s = objToStr(obj);
return s.includes('Object');
};
/**
* @type { import("../typings/UserJS.d.ts").isFN }
*/
const isFN = (obj) => {
const s = objToStr(obj);
return s.includes('Function');
};
/**
* @type { import("../typings/UserJS.d.ts").isNull }
*/
const isNull = (obj) => {
return Object.is(obj, null) || Object.is(obj, undefined);
};
/**
* @type { import("../typings/UserJS.d.ts").isBlank }
*/
const isBlank = (obj) => {
return (
(typeof obj === 'string' && Object.is(obj.trim(), '')) ||
((obj instanceof Set || obj instanceof Map) && Object.is(obj.size, 0)) ||
(Array.isArray(obj) && Object.is(obj.length, 0)) ||
(isObj(obj) && Object.is(Object.keys(obj).length, 0))
);
};
/**
* @type { import("../typings/UserJS.d.ts").isEmpty }
*/
const isEmpty = (obj) => {
return isNull(obj) || isBlank(obj);
};
/**
* @type { import("../typings/UserJS.d.ts").setObj }
*/
const setObj = (objA = {}, objB = {}) => {
objA = objA || {};
objB = objB || {};
for (const [key, value] of Object.entries(objA)) {
if (!hasOwn(objB, key)) {
objB[key] = value;
} else if (typeof value === 'object') {
setObj(value, objB[key]);
}
}
return objB;
};
/**
* @type { import("../typings/UserJS.d.ts").normalizeTarget }
*/
const normalizeTarget = (target, toQuery = true, root) => {
if (Object.is(target, null) || Object.is(target, undefined)) {
return [];
}
if (Array.isArray(target)) {
return target;
}
if (typeof target === 'string') {
return toQuery ? Array.from((root || document).querySelectorAll(target)) : [target];
}
if (isElem(target)) {
return [target];
}
return Array.from(target);
};
/**
* @type { import("../typings/UserJS.d.ts").ael }
*/
const ael = (el, type, listener, options = {}) => {
try {
for (const elem of normalizeTarget(el)) {
if (!elem) {
continue;
}
if (isMobile && type === 'click') {
elem.addEventListener('touchstart', listener);
continue;
}
elem.addEventListener(type, listener, options);
}
} catch (ex) {
err(ex);
}
};
/**
* @type { import("../typings/UserJS.d.ts").formAttrs }
*/
const formAttrs = (elem, attr = {}) => {
if (!elem) {
return elem;
}
for (const key in attr) {
if (typeof attr[key] === 'object') {
formAttrs(elem[key], attr[key]);
} else if (isFN(attr[key])) {
if (/^on/.test(key)) {
elem[key] = attr[key];
continue;
}
ael(elem, key, attr[key]);
} else if (key === 'class') {
elem.className = attr[key];
} else {
elem[key] = attr[key];
}
}
return elem;
};
/**
* @type { import("../typings/UserJS.d.ts").make }
*/
const make = (tagName, cname, attrs) => {
let el;
try {
const safe = safeSelf();
el = safe.createElement(tagName);
if (!isEmpty(cname)) {
if (typeof cname === 'string') {
el.className = cname;
} else if (isObj(cname)) {
formAttrs(el, attrs);
}
}
if (!isEmpty(attrs)) {
if (typeof attrs === 'string') {
el.textContent = attrs;
} else if (isObj(attrs)) {
formAttrs(el, attrs);
}
}
} catch (ex) {
err(ex);
}
return el;
};
/**
* @type { import("../typings/UserJS.d.ts").getGMInfo }
*/
const getGMInfo = () => {
if (isGM) {
if (isFN(GM.info)) {
return GM.info;
} else if (isFN(GM_info)) {
return GM_info;
}
}
return {
script: {
icon: '',
name: 'Magic Userscript+',
namespace: 'https://github.com/magicoflolis/Userscript-Plus',
updateURL: 'https://github.com/magicoflolis/Userscript-Plus/releases',
version: 'Bookmarklet',
bugs: 'https://github.com/magicoflolis/Magic-PH/issues/new/choose'
}
};
};
MU.info = getGMInfo();
// #endregion
/**
* Based on uBlock Origin by Raymond Hill (https://github.com/gorhill/uBlock)
*
* [uBlock Origin Reference](https://github.com/gorhill/uBlock/blob/master/src/js/dom.js)
*/
class dom {
/**
* @template { HTMLElement } T
* @param { T } target
* @param { string } attr
* @param { * } [value=undefined]
*/
static attr(target, attr, value = undefined) {
for (const elem of normalizeTarget(target)) {
if (value === undefined) {
return elem.getAttribute(attr);
}
if (value === null) {
elem.removeAttribute(attr);
} else {
elem.setAttribute(attr, value);
}
}
}
/**
* @template { HTMLElement } T
* @param { T } target
* @param { string } prop
* @param { * } [value=undefined]
* @returns { keyof T | void }
*/
static prop(target, prop, value = undefined) {
for (const elem of normalizeTarget(target)) {
if (value === undefined) {
return elem[prop];
}
elem[prop] = value;
}
}
/**
* @template { HTMLElement } T
* @param { T } target
* @param { string } text
*/
static text(target, text) {
const targets = normalizeTarget(target);
if (text === undefined) {
return targets.length !== 0 ? targets[0].textContent : undefined;
}
for (const elem of targets) {
elem.textContent = text;
}
}
}
dom.cl = class {
static add(target, name) {
if (Array.isArray(name)) {
for (const elem of normalizeTarget(target)) {
elem.classList.add(...name);
}
} else {
for (const elem of normalizeTarget(target)) {
elem.classList.add(name);
}
}
}
static remove(target, name) {
if (Array.isArray(name)) {
for (const elem of normalizeTarget(target)) {
elem.classList.remove(...name);
}
} else {
for (const elem of normalizeTarget(target)) {
elem.classList.remove(name);
}
}
}
static toggle(target, name, state) {
let r;
for (const elem of normalizeTarget(target)) {
r = elem.classList.toggle(name, state);
}
return r;
}
static has(target, name) {
for (const elem of normalizeTarget(target)) {
if (elem.classList.contains(name)) {
return true;
}
}
return false;
}
};
class Memorize {
constructor() {
this.store = new Map();
this.create('cfg', 'container', 'userjs');
}
/**
* @template { string } S
* @param { ...S } maps
* @returns { S | S[] }
*/
create(...maps) {
const resp = [];
for (const key of maps) {
if (this.store.has(key)) {
return this.store.get(key);
}
this.store.set(key, new Map());
resp.push(this.store.get(key));
}
return resp.length >= 2 ? resp : resp[0];
}
}
const memory = new Memorize();
const iconSVG = {
cfg: {
viewBox: '0 0 24 24',
html: ''
},
close: {
viewBox: '0 0 24 24',
html: ''
},
filter: {
viewBox: '0 0 24 24',
html: ''
},
fsClose: {
viewBox: '0 0 24 24',
html: ' '
},
fsOpen: {
viewBox: '0 0 24 24',
html: ' '
},
fullscreen: {
viewBox: '0 0 96 96',
html: ''
},
gf: {
viewBox: '0 0 510.4 510.4',
html: ''
},
gh: {
viewBox: '0 0 16 16',
html: ''
},
hide: {
viewBox: '0 0 24 24',
html: ' '
},
install: {
viewBox: '0 0 16 16',
html: ''
},
issue: {
viewBox: '0 0 24 24',
html: ''
},
nav: {
viewBox: '0 0 24 24',
html: ' '
},
plus: {
viewBox: '0 0 24 24',
html: ''
},
search: {
viewBox: '0 0 24 24',
html: ''
},
verified: {
viewBox: '0 0 56 56',
fill: 'currentColor',
stroke: 'currentColor',
html: ''
},
load(type, container) {
const safe = safeSelf();
const svgElem = safe.createElementNS('http://www.w3.org/2000/svg', 'svg');
for (const [k, v] of Object.entries(iconSVG[type])) {
if (k === 'html') {
continue;
}
svgElem.setAttributeNS(null, k, v);
}
if (typeof iconSVG[type].html === 'string') {
svgElem.innerHTML = iconSVG[type].html;
}
if (container) {
container.appendChild(svgElem);
return svgElem;
}
return svgElem.outerHTML;
}
};
/**
* @type { import("../typings/UserJS.d.ts").StorageSystem }
*/
const StorageSystem = {
getItem(key) {
return window.localStorage.getItem(key);
},
has(key) {
return !isNull(this.getItem(key));
},
setItem(key, value) {
window.localStorage.setItem(key, value);
},
remove(key) {
window.localStorage.removeItem(key);
},
async setValue(key, v) {
v = typeof v === 'string' ? v : JSON.stringify(v ?? {});
if (isGM) {
if (isFN(GM.setValue)) {
await GM.setValue(key, v);
} else if (isFN(GM_setValue)) {
GM_setValue(key, v);
}
} else {
this.setItem(`MUJS-${key}`, v);
}
},
async getValue(key, def = {}) {
try {
if (isGM) {
let GMType;
if (isFN(GM.getValue)) {
GMType = await GM.getValue(key, JSON.stringify(def));
} else if (isFN(GM_getValue)) {
GMType = GM_getValue(key, JSON.stringify(def));
}
if (!isNull(GMType)) {
return JSON.parse(GMType);
}
}
return this.has(`MUJS-${key}`)
? JSON.parse(this.getItem(`MUJS-${key}`))
: def;
} catch (ex) {
err(ex);
}
}
};
/**
* @type { import("../typings/UserJS.d.ts").Network }
*/
const Network = {
async req(url, method = 'GET', responseType = 'json', data = {}, useFetch = false) {
if (isEmpty(url)) {
throw new Error('"url" parameter is empty');
}
method = Network.bscStr(method, false);
responseType = Network.bscStr(responseType);
const params = {
method,
...data
};
if (params.hermes) {
delete params.hermes;
}
if (isGM && !useFetch) {
if (params.credentials) {
Object.assign(params, {
anonymous: false
});
if (Object.is(params.credentials, 'omit')) {
Object.assign(params, {
anonymous: true
});
}
delete params.credentials;
}
} else if (params.onprogress) {
delete params.onprogress;
}
return await new Promise((resolve, reject) => {
if (isGM && !useFetch) {
Network.xmlRequest({
url,
responseType,
...params,
onerror: reject,
onload: (r_1) => {
if (r_1.status !== 200) reject(new Error(`${r_1.status} ${url}`));
if (responseType.match(/basic/)) resolve(r_1);
resolve(r_1.response);
}
});
} else {
fetch(url, params)
.then((response_1) => {
if (!response_1.ok) reject(response_1);
const check = (str_2 = 'text') => {
return isFN(response_1[str_2]) ? response_1[str_2]() : response_1;
};
if (responseType.match(/buffer/)) {
resolve(check('arrayBuffer'));
} else if (responseType.match(/json/)) {
resolve(check('json'));
} else if (responseType.match(/text/)) {
resolve(check('text'));
} else if (responseType.match(/blob/)) {
resolve(check('blob'));
} else if (responseType.match(/formdata/)) {
resolve(check('formData'));
} else if (responseType.match(/clone/)) {
resolve(check('clone'));
} else if (responseType.match(/document/)) {
const domParser = new DOMParser();
const respTxt = check('text');
if (respTxt instanceof Promise) {
respTxt.then((txt) => {
const doc = domParser.parseFromString(txt, 'text/html');
resolve(doc);
});
} else {
const doc = domParser.parseFromString(respTxt, 'text/html');
resolve(doc);
}
} else {
resolve(response_1);
}
})
.catch(reject);
}
});
},
format(bytes, decimals = 2) {
if (Number.isNaN(bytes)) return '0 Bytes';
const k = 1024;
const dm = decimals < 0 ? 0 : decimals;
const i = Math.floor(Math.pow(bytes) / Math.log(k));
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${Network.sizes[i]}`;
},
sizes: ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'],
async xmlRequest(details) {
if (isGM) {
if (isFN(GM.xmlHttpRequest)) {
return GM.xmlHttpRequest(details);
} else if (isFN(GM_xmlhttpRequest)) {
return GM_xmlhttpRequest(details);
}
}
const safe = safeSelf();
return await new Promise((resolve, reject) => {
const req = new safe.XMLHttpRequest();
let method = 'GET';
let url = 'about:blank';
let body;
for (const [key, value] of Object.entries(details)) {
if (key === 'onload') {
req.addEventListener('load', () => {
if (isFN(value)) {
value(req);
}
resolve(req);
});
} else if (key === 'onerror') {
req.addEventListener('error', (evt) => {
if (isFN(value)) {
value(evt);
}
reject(evt);
});
} else if (key === 'onabort') {
req.addEventListener('abort', (evt) => {
if (isFN(value)) {
value(evt);
}
reject(evt);
});
} else if (key === 'onprogress') {
req.addEventListener('progress', value);
} else if (key === 'responseType') {
if (value === 'buffer') {
req.responseType = 'arraybuffer';
} else {
req.responseType = value;
}
} else if (key === 'method') {
method = value;
} else if (key === 'url') {
url = value;
} else if (key === 'body') {
body = value;
}
}
req.open(method, url);
if (isEmpty(req.responseType)) {
req.responseType = 'text';
}
if (body) {
req.send(body);
} else {
req.send();
}
});
},
bscStr(str = '', lowerCase = true) {
const txt = str[lowerCase ? 'toLowerCase' : 'toUpperCase']();
return txt.replaceAll(/\W/g, '');
}
};
/**
* @type { import("../typings/UserJS.d.ts").qs }
*/
const qs = (selector, root) => {
try {
return (root || document).querySelector(selector);
} catch (ex) {
err(ex);
}
return null;
};
/**
* @type { import("../typings/UserJS.d.ts").qsA }
*/
const qsA = (selectors, root) => {
try {
return (root || document).querySelectorAll(selectors);
} catch (ex) {
err(ex);
}
return [];
};
const sleazyRedirect = () => {
const { hostname } = winLocation;
const gfSite = /greasyfork\.org/.test(hostname);
if (!gfSite && cfg.sleazyredirect) {
return;
}
const otherSite = gfSite ? 'sleazyfork' : 'greasyfork';
if (!qs('span.sign-in-link')) {
return;
}
if (!/scripts\/\d+/.test(winLocation.href)) {
return;
}
if (
!qs('#script-info') &&
(otherSite == 'greasyfork' || qs('div.width-constraint>section>p>a'))
) {
const str = winLocation.href.replace(
/\/\/([^.]+\.)?(greasyfork|sleazyfork)\.org/,
'//$1' + otherSite + '.org'
);
if (isFN(winLocation.assign)) {
winLocation.assign(str);
} else {
winLocation.href = str;
}
}
};
const intersect = (a = [], b = []) => {
for (const v of a) {
if (b.includes(v)) {
return true;
}
}
for (const v of b) {
if (a.includes(v)) {
return true;
}
}
return false;
};
// #region Container
class initContainer {
constructor(url) {
this.remove = this.remove.bind(this);
this.webpage = strToURL(url || window.location);
this.host = this.getHost(this.webpage.host);
this.domain = this.getDomain(this.webpage.host);
this.ready = false;
this.injected = false;
this.shadowRoot = undefined;
this.supported = isFN(make('main-userjs').attachShadow);
this.frame = this.supported
? make('main-userjs', '', {
dataset: {
insertedBy: MU.info.script.name,
role: 'primary-container'
}
})
: make('iframe', 'mujs-iframe', {
dataset: {
insertedBy: MU.info.script.name,
role: 'primary-iframe'
},
loading: 'lazy',
src: 'about:blank',
style:
'position: fixed;bottom: 1rem;right: 1rem;height: 525px;width: 90%;margin: 0px 1rem;z-index: 100000000000000020 !important;',
onload: (iFrame) => {
/**
* @type { HTMLIFrameElement }
*/
const target = iFrame.target;
if (!target.contentDocument) {
return;
}
this.shadowRoot = target.contentDocument.documentElement;
this.ready = true;
dom.cl.add([this.shadowRoot, target.contentDocument.body], 'mujs-iframe');
}
});
if (this.supported) {
/**
* @type { ShadowRoot }
*/
this.shadowRoot = this.frame.attachShadow({
mode: 'open',
clonable: false,
delegatesfocus: false
});
this.ready = true;
}
this.cache = memory.store.get('container');
this.userjsCache = memory.store.get('userjs');
this.root = make('mujs-root');
this.unsaved = false;
this.isBlacklisted = false;
this.rebuild = false;
this.counters = {
total: 0
};
this.opacityMin = '0.15';
this.opacityMax = '1';
this.Timeout = class {
constructor() {
this.ids = [];
}
set(delay, reason) {
return new Promise((resolve, reject) => {
const id = setTimeout(() => {
Object.is(reason, null) || Object.is(reason, undefined) ? resolve() : reject(reason);
this.clear(id);
}, delay);
this.ids.push(id);
});
}
clear(...ids) {
this.ids = this.ids.filter((id) => {
if (ids.includes(id)) {
clearTimeout(id);
return false;
}
return true;
});
}
};
window.addEventListener('beforeunload', this.remove);
}
/**
* @param { Function } callback
* @param { Document } doc
*/
async inject(callback, doc) {
if (this.checkBlacklist(this.host)) {
err('Blacklisted website');
this.remove();
return;
}
if (!this.shadowRoot) {
return;
}
if (!doc) {
return;
}
while (this.ready === false) {
await new Promise((resolve) => requestAnimationFrame(resolve));
}
try {
doc.documentElement.appendChild(this.frame);
if (this.injected) {
return;
}
this.shadowRoot.append(this.root);
if (isNull(this.loadCSS(main_css, 'primary-stylesheet'))) {
throw new Error('Failed to initialize script!', { cause: 'loadCSS' });
}
this.injected = true;
if (isFN(callback)) {
callback.call(this, this.shadowRoot);
}
} catch (ex) {
err(ex);
this.remove();
}
}
remove() {
memory.store.clear();
if (this.frame) {
this.frame.remove();
}
}
async save() {
this.unsaved = false;
await StorageSystem.setValue('Config', cfg);
info('Saved:', cfg);
return cfg;
}
/**
* @param { string } css - CSS to inject
* @param { string } name - Name of stylesheet
* @return { HTMLStyleElement } Style element
*/
loadCSS(css, name = 'CSS') {
try {
if (typeof name !== 'string') {
throw new Error('"name" must be a typeof "string"', { cause: 'loadCSS' });
}
if (qs(`style[data-role="${name}"]`, this.root)) {
return qs(`style[data-role="${name}"]`, this.root);
}
if (typeof css !== 'string') {
throw new Error('"css" must be a typeof "string"', { cause: 'loadCSS' });
}
if (isBlank(css)) {
throw new Error(`"${name}" contains empty CSS string`, { cause: 'loadCSS' });
}
const parent = isEmpty(this.root.shadowRoot) ? this.root : this.root.shadowRoot;
if (isGM) {
let sty;
if (isFN(GM.addElement)) {
sty = GM.addElement(parent, 'style', {
textContent: css
});
} else if (isFN(GM_addElement)) {
sty = GM_addElement(parent, 'style', {
textContent: css
});
}
if (isElem(sty)) {
sty.dataset.insertedBy = MU.info.script.name;
sty.dataset.role = name;
return sty;
}
}
const sty = make('style', '', {
textContent: css,
dataset: {
insertedBy: MU.info.script.name,
role: name
}
});
parent.appendChild(sty);
return sty;
} catch (ex) {
err(ex);
}
return null;
}
checkBlacklist(str) {
str = str || this.host;
let blacklisted = false;
for (const b of normalizeTarget(cfg.blacklist)) {
if (typeof b === 'string') {
if (b.startsWith('userjs-')) {
const r = /userjs-(\w+)/.exec(b)[1];
const biList = builtinList[r];
if (isRegExp(biList)) {
if (!biList.test(str)) continue;
blacklisted = true;
} else if (isObj(biList) && biList.host === this.host) {
blacklisted = true;
}
}
} else if (isObj(b)) {
if (!b.enabled) {
continue;
}
if (b.regex === true) {
const reg = new RegExp(b.url, b.flags);
if (!reg.test(str)) continue;
blacklisted = true;
}
if (Array.isArray(b.url)) {
for (const c of b.url) {
if (!str.includes(c)) continue;
blacklisted = true;
}
}
if (!str.includes(b.url)) continue;
blacklisted = true;
}
}
this.isBlacklisted = blacklisted;
return blacklisted;
}
getInfo(url) {
const webpage = strToURL(url || this.webpage || window.location);
const host = this.getHost(webpage.host);
const domain = this.getDomain(webpage.host);
return {
domain,
host,
webpage
};
}
/**
* @template { string } S
* @param { S } str
* @returns { S }
*/
getHost(str = '') {
return str.split('.').splice(-2).join('.');
}
/**
* @template { string } S
* @param { S } str
* @returns { S }
*/
getDomain(str = '') {
return str.split('.').at(-2) ?? 'about:blank';
}
renderTheme(theme) {
theme = theme || cfg.theme;
if (theme === defcfg.theme) {
return;
}
const sty = this.root.style;
for (const [k, v] of Object.entries(theme)) {
const str = `--mujs-${k}`;
const prop = sty.getPropertyValue(str);
if (isEmpty(v)) {
theme[k] = prop;
}
if (prop === v) {
continue;
}
sty.removeProperty(str);
sty.setProperty(str, v);
}
}
}
const container = new initContainer();
// #endregion
// #region Primary Function
function primaryFN() {
try {
const mouseTimeout = new container.Timeout();
const frameTimeout = new container.Timeout();
const urlBar = make('input', 'mujs-url-bar', {
autocomplete: 'off',
spellcheck: false,
type: 'text',
placeholder: i18n$('search_placeholder')
});
const mainbtn = make('count-frame', 'mainbtn', {
textContent: '0'
});
const rateContainer = make('mujs-column', 'rate-container');
const footer = make('mujs-row', 'mujs-footer');
const tabbody = make('tbody');
const toolbar = make('mujs-toolbar');
const table = make('table');
const tabhead = make('thead');
const header = make('mujs-header');
const tbody = make('mujs-body');
const cfgpage = make('mujs-row', 'mujs-cfg hidden');
const countframe = make('mujs-column');
const btnframe = make('mujs-column', 'btn-frame');
const exBtn = make('mujs-column', 'mujs-sty-flex hidden');
const fsearch = make('mujs-btn', 'hidden');
const exportCFG = make('mujs-btn', 'mujs-export', {
textContent: i18n$('export_config'),
dataset: {
command: 'export-cfg'
}
});
const importCFG = make('mujs-btn', 'mujs-import', {
textContent: i18n$('import_config'),
dataset: {
command: 'import-cfg'
}
});
const exportTheme = make('mujs-btn', 'mujs-export', {
textContent: i18n$('export_theme'),
dataset: {
command: 'export-theme'
}
});
const importTheme = make('mujs-btn', 'mujs-import', {
textContent: i18n$('import_theme'),
dataset: {
command: 'import-theme'
}
});
const btnHandles = make('mujs-column', 'btn-handles');
const btnHide = make('mujs-btn', 'hide-list', {
title: i18n$('min'),
innerHTML: iconSVG.load('hide'),
dataset: {
command: 'hide-list'
}
});
const btnfullscreen = make('mujs-btn', 'fullscreen', {
title: i18n$('max'),
innerHTML: iconSVG.load('fullscreen'),
dataset: {
command: 'fullscreen'
}
});
const main = make('mujs-main', 'hidden');
const urlContainer = make('mujs-url');
const closebtn = make('mujs-btn', 'close', {
title: i18n$('close'),
innerHTML: iconSVG.load('close'),
dataset: {
command: 'close'
}
});
const btncfg = make('mujs-btn', 'settings hidden', {
title: 'Settings',
innerHTML: iconSVG.load('cfg'),
dataset: {
command: 'settings'
}
});
const btnhome = make('mujs-btn', 'github hidden', {
title: `GitHub (v${
MU.info.script.version.includes('.') || MU.info.script.version.includes('Book')
? MU.info.script.version
: MU.info.script.version.slice(0, 5)
})`,
innerHTML: iconSVG.load('gh'),
dataset: {
command: 'open-tab',
webpage: MU.info.script.namespace
}
});
const btnissue = make('mujs-btn', 'issue hidden', {
innerHTML: iconSVG.load('issue'),
title: i18n$('issue'),
dataset: {
command: 'open-tab',
webpage: MU.info.script.bugs
}
});
const btngreasy = make('mujs-btn', 'greasy hidden', {
title: 'Greasy Fork',
innerHTML: iconSVG.load('gf'),
dataset: {
command: 'open-tab',
webpage: 'https://greasyfork.org/scripts/421603'
}
});
const btnnav = make('mujs-btn', 'nav', {
title: 'Navigation',
innerHTML: iconSVG.load('nav'),
dataset: {
command: 'navigation'
}
});
const mainframe = make('mu-js', 'mainframe', {
style: `opacity: ${container.opacityMin};`
});
btnHandles.append(btnHide, btnfullscreen, closebtn);
btnframe.append(btnhome, btngreasy, btnissue, btncfg, btnnav);
toolbar.append(btnHandles);
urlContainer.append(urlBar);
header.append(urlContainer, rateContainer, countframe, btnframe);
tbody.append(table, cfgpage);
main.append(toolbar, header, tbody, footer);
mainframe.append(mainbtn);
exBtn.append(importCFG, importTheme, exportCFG, exportTheme);
header.append(exBtn);
container.root.append(mainframe, main);
for (const engine of cfg.engines) {
container.counters[engine.name] = 0;
if (!engine.enabled) {
continue;
}
const counter = make('count-frame', '', {
dataset: {
counter: engine.name
},
title: engine.url,
textContent: '0'
});
countframe.append(counter);
}
class Tabs {
constructor() {
this.Tab = new Map();
this.blank = 'about:blank';
this.protocal = 'mujs:';
this.protoReg = new RegExp(`${this.protocal}(.+)`);
this.el = {
add: make('mujs-addtab', '', {
textContent: '+',
dataset: {
command: 'new-tab'
}
}),
head: make('mujs-tabs')
};
this.el.head.append(this.el.add);
toolbar.append(this.el.head);
}
hasTab(...params) {
for (const p of params) {
if (!this.Tab.has(p)) {
return false;
}
const content = normalizeTarget(this.Tab.get(p)).filter((t) => p === t.dataset.host);
if (isBlank(content)) {
return false;
}
}
return true;
}
storeTab(host) {
const h = host ?? this.blank;
if (!this.Tab.has(h)) {
this.Tab.set(h, new Set());
}
return this.Tab.get(h);
}
cache(host, ...tabs) {
const h = host ?? this.blank;
const tabCache = this.storeTab(h);
for (const t of normalizeTarget(tabs)) {
if (tabCache.has(t)) {
continue;
}
tabCache.add(t);
}
this.Tab.set(h, tabCache);
return tabCache;
}
mujs(host) {
if (!host.startsWith(this.protocal)) {
return;
}
const type = host.match(this.protoReg)[1];
if (type === 'settings') {
dom.cl.remove([cfgpage, exBtn], 'hidden');
dom.cl.add(table, 'hidden');
if (!container.supported) {
dom.attr(container.frame, 'style', 'height: 100%;');
}
}
}
active(tab, build = true) {
for (const t of normalizeTarget(tab, false)) {
dom.cl.add([cfgpage, exBtn], 'hidden');
dom.cl.remove(table, 'hidden');
dom.cl.remove(qsA('mujs-tab', this.el.head), 'active');
dom.cl.add(t, 'active');
if (!build) {
continue;
}
const host = t.dataset.host ?? this.blank;
if (host === this.blank) {
MUJS.refresh();
} else if (host.startsWith(this.protocal)) {
this.mujs(host);
} else {
buildlist(host);
}
}
}
/** @param { HTMLElement } tab */
close(tab) {
for (const t of normalizeTarget(tab, false)) {
const host = t.dataset.host;
if (container.cache.has(host)) {
container.cache.delete(host);
}
if (dom.cl.has(t, 'active')) {
MUJS.refresh();
}
const sibling = t.previousElementSibling ?? t.nextElementSibling;
if (sibling) {
if (sibling.dataset.command !== 'new-tab') {
this.active(sibling);
}
}
if (this.Tab.has(host)) {
this.Tab.delete(host);
}
t.remove();
}
}
create(host = undefined) {
if (typeof host === 'string') {
if (host.startsWith(this.protocal) && this.hasTab(host)) {
this.active(this.Tab.get(host));
return;
}
const content = normalizeTarget(this.storeTab(host)).filter(
(t) => host === t.dataset.host
);
if (!isEmpty(content)) {
return;
}
}
const tab = make('mujs-tab', '', {
dataset: {
command: 'switch-tab'
},
style: `order: ${this.el.head.childElementCount};`
});
const tabClose = make('mu-js', '', {
dataset: {
command: 'close-tab'
},
title: i18n$('close'),
textContent: 'X'
});
const tabHost = make('mujs-host');
tab.append(tabHost, tabClose);
this.el.head.append(tab);
this.active(tab, false);
this.cache(host, tab);
if (isNull(host)) {
MUJS.refresh();
urlBar.placeholder = i18n$('newTab');
tab.dataset.host = this.blank;
tabHost.title = i18n$('newTab');
tabHost.textContent = i18n$('newTab');
} else if (host.startsWith(this.protocal)) {
const type = host.match(this.protoReg)[1];
tab.dataset.host = host || container.host;
tabHost.title = type || tab.dataset.host;
tabHost.textContent = tabHost.title;
this.mujs(host);
} else {
tab.dataset.host = host || container.host;
tabHost.title = host || container.host;
tabHost.textContent = tabHost.title;
}
return tab;
}
}
const tab = new Tabs();
const cfgMap = memory.store.get('cfg');
const rebuildCfg = () => {
for (const engine of cfg.engines) {
if (cfgMap.has(engine.name)) {
const inp = cfgMap.get(engine.name);
inp.checked = engine.enabled;
if (engine.name === 'github') {
const txt = cfgMap.get('github-token');
dom.prop(txt, 'value', engine.token);
}
}
}
for (const [k, v] of Object.entries(cfg)) {
if (typeof v === 'boolean') {
if (cfgMap.has(k)) {
const inp = cfgMap.get(k);
if (inp.type === 'checkbox') {
inp.checked = v;
} else {
dom.prop(inp, 'value', v);
}
}
}
}
dom.prop(cfgMap.get('blacklist'), 'value', JSON.stringify(cfg.blacklist, null, ' '));
dom.prop(cfgMap.get('theme'), 'value', JSON.stringify(cfg.theme, null, ' '));
container.renderTheme(cfg.theme);
};
const doInstallProcess = async (installLink) => {
if (isFN(winLocation.assign)) {
winLocation.assign(installLink.href);
} else {
winLocation.href = installLink.href;
}
installLink.remove();
await init();
};
// #region Main event handlers
ael(main, 'click', async (evt) => {
try {
/** @type { HTMLElement } */
const target = evt.target.closest('[data-command]');
if (!target) {
return;
}
const dataset = target.dataset;
const cmd = dataset.command;
if (cmd === 'install-script' && dataset.userjs) {
const dlBtn = make('a', '', {
onclick(evt) {
evt.preventDefault();
doInstallProcess(evt.target);
}
});
dlBtn.href = dataset.userjs;
dlBtn.click();
} else if (cmd === 'open-tab' && dataset.webpage) {
if (isGM) {
if (isFN(GM.openInTab)) {
return GM.openInTab(dataset.webpage);
} else if (isFN(GM_openInTab)) {
return GM_openInTab(dataset.webpage, {
active: true,
insert: true
});
}
}
return window.open(dataset.webpage, '_blank');
} else if (cmd === 'navigation') {
if (dom.cl.has(btngreasy, 'hidden')) {
dom.cl.remove([btncfg, btngreasy, btnhome, btnissue], 'hidden');
} else {
dom.cl.add([btncfg, btngreasy, btnhome, btnissue], 'hidden');
}
} else if (cmd === 'list-description') {
const arr = [];
const ignoreTags = new Set(['TD', 'MUJS-A', 'MU-JS']);
for (const node of target.parentElement.childNodes) {
if (ignoreTags.has(node.tagName)) {
continue;
}
if (node.tagName === 'TEXTAREA' && isEmpty(node.value)) {
continue;
}
arr.push(node);
}
if (target.nextElementSibling) {
arr.push(target.nextElementSibling);
if (target.nextElementSibling.nextElementSibling) {
arr.push(target.nextElementSibling.nextElementSibling);
}
}
if (dom.cl.has(arr[0], 'hidden')) {
dom.cl.remove(arr, 'hidden');
} else {
dom.cl.add(arr, 'hidden');
}
} else if (cmd === 'close') {
container.remove();
} else if (cmd === 'show-filter') {
dom.cl.toggle(fsearch, 'hidden');
} else if (cmd === 'fullscreen') {
if (dom.cl.has(btnfullscreen, 'expanded')) {
dom.cl.remove([btnfullscreen, main], 'expanded');
dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('fsOpen'));
} else {
dom.cl.add([btnfullscreen, main], 'expanded');
dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('fsClose'));
}
} else if (cmd === 'hide-list') {
dom.cl.add(main, 'hidden');
dom.cl.remove(mainframe, 'hidden');
timeoutFrame();
} else if (cmd === 'save') {
if (!isNull(ghMsg)) {
ghMsg = null;
container.rebuild = true;
dom.prop(rateContainer, 'innerHTML', '');
}
if (!dom.prop(target, 'disabled')) {
container.save();
sleazyRedirect();
if (container.rebuild) {
container.cache.clear();
buildlist();
}
container.unsaved = false;
container.rebuild = false;
}
} else if (cmd === 'reset') {
cfg = defcfg;
dom.cl.remove(mainframe, 'error');
if (qs('.error', footer)) {
for (const elem of normalizeTarget(qsA('.error', footer))) {
elem.remove();
}
}
if (container.oldBlacklist) {
delete container.oldBlacklist;
}
container.unsaved = true;
container.rebuild = true;
rebuildCfg();
} else if (cmd === 'settings') {
if (container.unsaved) {
MUJS.showError('Unsaved changes');
}
tab.create('mujs:settings');
container.rebuild = false;
} else if (cmd === 'new-tab') {
tab.create();
} else if (cmd === 'switch-tab') {
tab.active(target);
} else if (cmd === 'close-tab' && target.parentElement) {
tab.close(target.parentElement);
} else if (cmd === 'download-userjs') {
if (!container.userjsCache.has(+dataset.userjs)) {
return;
}
const dataUserJS = container.userjsCache.get(+dataset.userjs);
const txt = await reqCode(dataUserJS);
if (typeof txt !== 'string') {
return;
}
const makeUserJS = new Blob([txt], { type: 'text/plain' });
const dlBtn = make('a', 'mujs_Downloader');
dlBtn.href = URL.createObjectURL(makeUserJS);
dlBtn.download = `${dataset.userjsName ?? dataset.userjs}.user.js`;
dlBtn.click();
URL.revokeObjectURL(dlBtn.href);
dlBtn.remove();
} else if (cmd === 'load-userjs') {
if (!container.userjsCache.has(+dataset.userjs)) {
return;
}
const codeArea = qs('textarea', target.parentElement.parentElement);
if (!isEmpty(codeArea.value)) {
dom.cl.toggle(codeArea, 'hidden');
return;
}
const dataUserJS = container.userjsCache.get(+dataset.userjs);
const txt = await reqCode(dataUserJS);
if (typeof txt !== 'string') {
return;
}
codeArea.value = txt;
dom.cl.remove(codeArea, 'hidden');
const apTo = (name, elem) => {
if (isEmpty(dataUserJS[name])) {
const el = make('mujs-a', '', {
textContent: i18n$('listing_none')
});
elem.append(el);
if (name === 'antifeatures') {
dom.cl.add(elem, 'hidden');
}
} else {
for (const c of dataUserJS[name]) {
if (isObj(c)) {
const el = make('mujs-a', '', {
textContent: c.text
});
if (c.domain) {
el.dataset.command = 'open-tab';
el.dataset.webpage = `https://${c.text}`;
}
elem.append(el);
} else {
const el = make('mujs-a', '', {
textContent: c
});
elem.append(el);
}
}
if (name === 'antifeatures') {
dom.cl.remove(elem, 'hidden');
}
}
};
const matchElem = qs(
'[data-type="match-urls"] > .mujs-grants',
target.parentElement.parentElement
);
const grantElem = qs(
'[data-type="grants"] > .mujs-grants',
target.parentElement.parentElement
);
const afElem = qs(
'[data-type="antifeatures"] > .mujs-grants',
target.parentElement.parentElement
);
const sizeElem = qs(
'[data-type="size"] > .mujs-grants',
target.parentElement.parentElement
);
dom.prop([matchElem, grantElem, afElem, sizeElem], 'innerHTML', '');
apTo('code_match', matchElem);
apTo('code_grant', grantElem);
apTo('antifeatures', afElem);
apTo('code_size', sizeElem);
} else if (/export-/.test(cmd)) {
const str = JSON.stringify(cmd === 'export-cfg' ? cfg : cfg.theme, null, ' ');
const bytes = new TextEncoder().encode(str);
const blob = new Blob([bytes], { type: 'application/json;charset=utf-8' });
const dlBtn = make('a', 'mujs-exporter', {
href: URL.createObjectURL(blob),
download: `Magic_Userscript_${cmd === 'export-cfg' ? 'config' : 'theme'}.json`
});
dlBtn.click();
URL.revokeObjectURL(dlBtn.href);
} else if (/import-/.test(cmd)) {
if (qs('input', target.parentElement)) {
qs('input', target.parentElement).click();
return;
}
const inpJSON = make('input', 'hidden', {
type: 'file',
accept: '.json',
onchange(evt) {
try {
[...evt.target.files].forEach((file) => {
const reader = new FileReader();
reader.readAsText(file);
reader.onload = () => {
const result = JSON.parse(reader.result);
if (result.blacklist) {
log(`Imported config: { ${file.name} }`, result);
cfg = result;
container.unsaved = true;
container.rebuild = true;
rebuildCfg();
container.save();
sleazyRedirect();
container.cache.clear();
buildlist();
container.unsaved = false;
container.rebuild = false;
} else {
log(`Imported theme: { ${file.name} }`, result);
cfg.theme = result;
container.renderTheme(cfg.theme);
}
inpJSON.remove();
};
reader.onerror = () => {
MUJS.showError(reader.error);
inpJSON.remove();
};
});
} catch (ex) {
MUJS.showError(ex);
inpJSON.remove();
}
}
});
target.parentElement.append(inpJSON);
inpJSON.click();
}
} catch (ex) {
MUJS.showError(ex);
}
});
ael(main, 'auxclick', (evt) => {
if (evt.button !== 1) {
return;
}
/** @type { HTMLElement } */
const target = evt.target.closest('[data-command]');
if (!target) {
return;
}
const dataset = target.dataset;
const cmd = dataset.command;
if (cmd === 'switch-tab' || cmd === 'close-tab') {
tab.close(target);
} else if (cmd === 'new-tab') {
tab.create();
}
});
if (!isMobile) {
const fade = async (target, type) => {
if (type === 'mouseenter') {
frameTimeout.clear(...frameTimeout.ids);
mouseTimeout.clear(...mouseTimeout.ids);
target.style.opacity = container.opacityMax;
} else if (type === 'mouseleave') {
await mouseTimeout.set(cfg.time);
target.style.opacity = container.opacityMin;
}
};
for (const e of ['mouseenter', 'mouseleave']) {
ael(main, e, (evt) => {
evt.preventDefault();
evt.stopPropagation();
fade(evt.target, evt.type);
});
}
}
// #endregion
const ContainerHandler = class {
constructor() {
this.showError = this.showError.bind(this);
}
checkBlacklist(str) {
if (container.checkBlacklist(str)) {
this.showError('Blacklisted');
timeoutFrame();
return true;
}
return false;
}
updateCounter(count, engine) {
container.counters[engine.name] += count;
container.counters.total += count;
this.updateCounters();
}
updateCounters() {
for (const [k, v] of Object.entries(container.counters)) {
if (k === 'total') {
continue;
}
if (qs(`count-frame[data-counter="${k}"]`, countframe)) {
dom.text(qs(`count-frame[data-counter="${k}"]`, countframe), v);
}
}
dom.text(mainbtn, container.counters.total);
}
makeError(...ex) {
const safe = safeSelf();
const error = make('mu-js', 'error');
let str = '';
for (const e of ex) {
str += `${typeof e === 'string' ? e : `${e.cause ? `[${e.cause}] ` : ''}${e.message} ${e.stack ?? ''}`}\n`;
if (isObj(e)) {
if (e.notify) {
dom.cl.add(mainframe, 'error');
}
}
}
error.appendChild(safe.createTextNode(str));
footer.append(error);
}
showError(...ex) {
err(...ex);
this.makeError(...ex);
}
refresh() {
urlBar.placeholder = i18n$('newTab');
container.counters.total = 0;
for (const engine of cfg.engines) {
container.counters[engine.name] = 0;
}
this.updateCounters();
dom.prop([tabbody, rateContainer, footer], 'innerHTML', '');
}
};
const MUJS = new ContainerHandler();
const timeoutFrame = async (time) => {
time = time ?? cfg.time;
frameTimeout.clear(...frameTimeout.ids);
if (dom.cl.has(mainframe, 'hidden')) {
return;
}
if (typeof time === 'number' && !Number.isNaN(time)) {
await frameTimeout.set(container.isBlacklisted ? time / 2 : time);
} else {
await frameTimeout.set(10000);
}
container.remove();
return frameTimeout.clear(...frameTimeout.ids);
};
/**
* @param { string } code
*/
const get_meta_block = (code) => {
if (isEmpty(code)) {
return null;
}
const start_block = code.indexOf(META_START_COMMENT);
if (isNull(start_block)) {
return null;
}
const end_block = code.indexOf(META_END_COMMENT, start_block);
if (isNull(end_block)) {
return null;
}
return code.substring(start_block + META_START_COMMENT.length, end_block);
};
/**
* @param { string } code
*/
const parse_meta = (code) => {
if (isEmpty(code)) {
return null;
}
const meta = {};
const meta_block = get_meta_block(code);
const meta_block_map = new Map();
for (const meta_line of meta_block.split('\n')) {
const meta_match = meta_line.match(/\/\/\s+@([a-zA-Z:-]+)\s+(.*)/);
if (isNull(meta_match)) {
continue;
}
const key = meta_match[1].trim();
const value = meta_match[2].trim();
if (!meta_block_map.has(key)) {
meta_block_map.set(key, []);
}
const meta_map = meta_block_map.get(key);
meta_map.push(value);
meta_block_map.set(key, meta_map);
}
for (const [key, value] of meta_block_map) {
if (value.length > 1) {
meta[key] = value;
} else {
meta[key] = value[0];
}
}
return meta;
};
/**
* @template { string } S
* @param { S } code
* @returns { S[] }
*/
const calculate_applies_to_names = (code) => {
if (isEmpty(code)) {
return null;
}
const meta = parse_meta(code);
let patterns = [];
for (const [k, v] of Object.entries(meta)) {
if (/include|match/.test(k)) {
if (Array.isArray(v)) {
patterns = patterns.concat(v);
} else {
patterns = patterns.concat([v]);
}
}
}
if (isEmpty(patterns)) {
return [];
}
if (intersect(patterns, APPLIES_TO_ALL_PATTERNS)) {
return [];
}
const name_set = new Set();
const addObj = (obj) => {
if (name_set.has(obj)) {
return;
}
name_set.add(obj);
};
for (let p of patterns) {
try {
const original_pattern = p;
let pre_wildcards = [];
if (p.match(/^\/(.*)\/$/)) {
pre_wildcards = [p];
} else {
let m = p.match(/^\*(https?:.*)/i);
if (!isNull(m)) {
p = m[1];
}
p = p
.replace(/^\*:/i, 'http:')
.replace(/^\*\/\//i, 'http://')
.replace(/^http\*:/i, 'http:')
.replace(/^(https?):([^/])/i, '$1://$2');
m = p.match(/^([a-z]+:\/\/)\*\.?([a-z0-9-]+(?:.[a-z0-9-]+)+.*)/i);
if (!isNull(m)) {
p = m[1] + m[2];
}
m = p.match(/^\*\.?([a-z0-9-]+\.[a-z0-9-]+.*)/i);
if (!isNull(m)) {
p = `http://${m[1]}`;
}
m = p.match(/^http\*(?:\/\/)?\.?((?:[a-z0-9-]+)(?:\.[a-z0-9-]+)+.*)/i);
if (!isNull(m)) {
p = `http://${m[1]}`;
}
m = p.match(/^([a-z]+:\/\/([a-z0-9-]+(?:\.[a-z0-9-]+)*\.))\*(.*)/);
if (!isNull(m)) {
if (m[2].match(/A([0-9]+\.){2,}z/)) {
p = `${m[1]}tld${m[3]}`;
pre_wildcards = [p.split('*')[0]];
} else {
pre_wildcards = [p];
}
} else {
pre_wildcards = [p];
}
}
for (const pre_wildcard of pre_wildcards) {
try {
const urlObj = new URL(pre_wildcard);
const { host } = urlObj;
if (isNull(host)) {
addObj({ text: original_pattern, domain: false, tld_extra: false });
} else if (!host.includes('.') && host.includes('*')) {
addObj({ text: original_pattern, domain: false, tld_extra: false });
} else if (host.endsWith('.tld')) {
for (let i = 0; i < TLD_EXPANSION.length; i++) {
const tld = TLD_EXPANSION[i];
addObj({
text: host.replace(/tld$/i, tld),
domain: true,
tld_extra: i != 0
});
}
} else if (host.endsWith('.')) {
addObj({
text: host.slice(0, -1),
domain: true,
tld_extra: false
});
} else {
addObj({
text: host,
domain: true,
tld_extra: false
});
}
// eslint-disable-next-line no-unused-vars
} catch (ex) {
addObj({ text: original_pattern, domain: false, tld_extra: false });
}
}
} catch (ex) {
err(ex);
}
}
return [...name_set];
};
const reqCode = async (obj = {}, translate = false) => {
if (obj.code_data) {
return obj.code_data;
}
/** @type { string } */
const code = await Network.req(obj.code_url, 'GET', 'text').catch(MUJS.showError);
if (typeof code !== 'string') {
return;
}
Object.assign(obj, {
code_data: code,
code_size: [Network.format(code.length)],
code_match: [],
code_grant: [],
antifeatures: []
});
const grantSet = new Set();
const afSet = new Set();
const meta = parse_meta(code);
const applies_to_names = calculate_applies_to_names(code);
for (const [key, value] of Object.entries(meta)) {
if (/grant/.test(key)) {
for (const v of normalizeTarget(value, false)) {
if (grantSet.has(v)) {
continue;
}
grantSet.add(v);
}
} else if (/antifeature/.test(key)) {
for (const v of normalizeTarget(value, false)) {
if (afSet.has(v)) {
continue;
}
afSet.add(v);
}
}
}
Object.assign(obj, {
code_match: applies_to_names,
code_grant: [...grantSet],
antifeatures: [...afSet]
});
if (translate) {
const headers = code.match(/\/\/\s*@[\w][\s\S]+/g);
if (isNull(headers)) {
return code;
}
for (const lng of language.cache) {
const findName = new RegExp(`//\\s*@name:${lng}\\s*(.*)`, 'gi').exec(headers[0]);
const findDesc = new RegExp(`//\\s*@description:${lng}\\s*(.*)`, 'gi').exec(headers[0]);
if (!isNull(findName)) {
Object.assign(obj, {
name: findName[1],
translated: true
});
}
if (!isNull(findDesc)) {
Object.assign(obj, {
description: findDesc[1],
translated: true
});
}
}
}
return code;
};
const template = {
id: 0,
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: ''
}
]
};
const mergeTemplate = (obj = {}) => {
for (const key in template) {
if (hasOwn(obj, key)) continue;
obj[key] = template[key];
}
return obj;
};
const mkList = (txt = '', obj = {}) => {
if (!obj.root || !obj.type) {
return;
}
const { root, type } = obj;
const list = obj.list ?? [];
const appliesTo = make('mu-js', 'mujs-list', {
textContent: `${txt}: `
});
const applyList = make('mu-js', 'mujs-grants');
const ujsURLs = make('mujs-column', 'mujs-list', {
dataset: {
el: 'matches',
type
}
});
ujsURLs.append(appliesTo, applyList);
root.append(ujsURLs);
if (isEmpty(list)) {
const elem = make('mujs-a', '', {
textContent: i18n$('listing_none')
});
applyList.append(elem);
if (type === 'antifeatures') {
dom.cl.add(ujsURLs, 'hidden');
}
return;
}
for (const c of list) {
if (isObj(c)) {
const elem = make('mujs-a', '', {
textContent: c.text
});
if (c.domain) {
elem.dataset.command = 'open-tab';
elem.dataset.webpage = `https://${c.text}`;
}
applyList.append(elem);
} else {
const elem = make('mujs-a', '', {
textContent: c
});
applyList.append(elem);
}
}
if (type === 'antifeatures') {
dom.cl.remove(ujsURLs, 'hidden');
}
};
const toLocaleDate = (str = '') => {
return new Intl.DateTimeFormat(navigator.language).format(new Date(str));
};
// #region Create UserJS
const createjs = (ujs, engine) => {
for (const key in template) {
if (hasOwn(ujs, key)) continue;
ujs[key] = template[key];
}
// Lets not add this UserJS to the list
if (ujs.id === 421603) {
return;
}
if (!container.userjsCache.has(ujs.id)) {
container.userjsCache.set(ujs.id, ujs);
}
const eframe = make('td', 'install-btn');
const uframe = make('td', 'mujs-uframe');
const fdaily = make('td', 'mujs-list', {
textContent: ujs.daily_installs
});
const fupdated = make('td', 'mujs-list', {
textContent: toLocaleDate(ujs.code_updated_at)
});
const fname = make('td', 'mujs-name');
const fmore = make('mujs-column', 'mujs-list hidden', {
dataset: {
el: 'more-info'
}
});
const fBtns = make('mujs-column', 'mujs-list hidden');
const jsInfo = make('mujs-row', 'mujs-list');
const jsInfoB = make('mujs-row', 'mujs-list');
const ratings = make('mujs-column', 'mujs-list');
const ftitle = make('mujs-a', 'mujs-homepage', {
textContent: ujs.name,
title: ujs.url,
dataset: {
command: 'open-tab',
webpage: ujs.url
}
});
const fver = make('mu-js', 'mujs-list', {
textContent: `${i18n$('version_number')}: ${ujs.version}`
});
const fcreated = make('mu-js', 'mujs-list', {
textContent: `${i18n$('created_date')}: ${toLocaleDate(ujs.created_at)}`
});
const flicense = make('mu-js', 'mujs-list', {
title: ujs.license ?? i18n$('no_license'),
textContent: `${i18n$('license')}: ${ujs.license ?? i18n$('no_license')}`,
style:
'text-overflow: ellipsis; overflow: hidden; white-space: nowrap; width: fit-content; max-width: 20em;'
});
const ftotal = make('mu-js', 'mujs-list', {
textContent: `${i18n$('total_installs')}: ${ujs.total_installs}`
});
const fratings = make('mu-js', 'mujs-list', {
title: i18n$('ratings'),
textContent: `${i18n$('ratings')}:`
});
const fgood = make('mu-js', 'mujs-list mujs-ratings', {
title: i18n$('good'),
textContent: ujs.good_ratings,
dataset: {
el: 'good'
}
});
const fok = make('mu-js', 'mujs-list mujs-ratings', {
title: i18n$('ok'),
textContent: ujs.ok_ratings,
dataset: {
el: 'ok'
}
});
const fbad = make('mu-js', 'mujs-list mujs-ratings', {
title: i18n$('bad'),
textContent: ujs.bad_ratings,
dataset: {
el: 'bad'
}
});
const fdesc = make('mu-js', 'mujs-list mujs-pointer', {
title: ujs.description,
textContent: ujs.description,
dataset: {
command: 'list-description'
}
});
const scriptInstall = make('mu-jsbtn', 'install', {
innerHTML: `${iconSVG.load('install')} ${i18n$('install')}`,
title: `${i18n$('install')} "${ujs.name}"`,
dataset: {
command: 'install-script',
userjs: ujs.code_url
}
});
const scriptDownload = make('mu-jsbtn', '', {
innerHTML: `${iconSVG.load('install')} ${i18n$('saveFile')}`,
dataset: {
command: 'download-userjs',
userjs: ujs.id,
userjsName: ujs.name
}
});
const tr = make('tr', 'frame', {
dataset: {
scriptId: ujs.id
}
});
const codeArea = make('textarea', 'code-area hidden', {
dataset: {
name: 'code'
},
rows: '10',
autocomplete: false,
spellcheck: false,
wrap: 'soft'
});
const loadCode = make('mu-jsbtn', '', {
innerHTML: `${iconSVG.load('search')} ${i18n$('preview_code')}`,
dataset: {
command: 'load-userjs',
userjs: ujs.id
}
});
if (engine) {
tr.dataset.engine = engine;
if (!engine.includes('fork') && cfg.recommend.others && goodUserJS.includes(ujs.url)) {
tr.dataset.good = 'upsell';
}
}
for (const u of ujs.users) {
const user = make('mujs-a', '', {
innerHTML: u.name,
title: u.url,
dataset: {
command: 'open-tab',
webpage: u.url
}
});
if (cfg.recommend.author && u.id === authorID) {
tr.dataset.author = 'upsell';
dom.prop(user, 'innerHTML', `${u.name} ${iconSVG.load('verified')}`);
}
uframe.append(user);
}
if (engine.includes('fork') && cfg.recommend.others && goodUserJS.includes(ujs.id)) {
tr.dataset.good = 'upsell';
}
eframe.append(scriptInstall);
ratings.append(fratings, fgood, fok, fbad);
jsInfo.append(ftotal, ratings, fver, fcreated);
mkList('Size', {
list: ujs.code_size,
type: 'size',
root: jsInfo
});
jsInfoB.append(flicense);
mkList(i18n$('antifeatures'), {
list: ujs.antifeatures,
type: 'antifeatures',
root: jsInfoB
});
mkList(i18n$('applies_to'), {
list: ujs.code_match,
type: 'match-urls',
root: jsInfoB
});
mkList('@grant', {
list: ujs.code_grant,
type: 'grants',
root: jsInfoB
});
fmore.append(jsInfo, jsInfoB);
fBtns.append(scriptDownload, loadCode);
fname.append(ftitle, fdesc, fmore, fBtns, codeArea);
if (ujs.code_data) {
codeArea.value = ujs.code_data;
}
for (const e of [fname, uframe, fdaily, fupdated, eframe]) {
tr.append(e);
}
tabbody.append(tr);
};
// #endregion
//#region Build List
const buildlist = async (host = undefined) => {
try {
if (isEmpty(host)) {
host = container.host;
}
if (container.oldBlacklist) {
MUJS.makeError({ message: i18n$('legacy'), cause: 'Blacklist outdated', notify: true });
return;
}
if (MUJS.checkBlacklist(host)) {
return;
}
if (!isNull(ghMsg)) {
const txt = make('mujs-row', 'legacy-config', {
textContent: ghMsg
});
rateContainer.append(txt);
return;
}
MUJS.refresh();
if (!container.cache.has(host)) {
const engineTemplate = {};
for (const engine of cfg.engines) {
engineTemplate[engine.name] = [];
}
container.cache.set(host, engineTemplate);
}
const isSupported = (name) => {
for (const [k, v] of Object.entries(engineUnsupported)) {
if (k !== name) {
continue;
}
if (v.includes(host)) {
return false;
}
}
return true;
};
const engines = cfg.engines.filter((e) => e.enabled && isSupported(e.name));
// Fully hide when on a unsupported site.
if (isEmpty(engines)) {
container.opacityMin = '0';
mainframe.style.opacity = container.opacityMin;
return;
}
const cache = container.cache.get(host);
const customRecords = [];
info('Building list', { cache, MUJS, engines });
const initUserJS = async (ujs, engine) => {
if (cfg.codePreview && !ujs.code_data) {
await reqCode(ujs);
}
createjs(ujs, engine);
};
for (const engine of engines) {
const cEngine = cache[`${engine.name}`];
if (!isEmpty(cEngine)) {
for (const ujs of cEngine) {
createjs(ujs, engine.name);
}
MUJS.updateCounter(cEngine.length, engine);
continue;
}
const forkFN = async (data) => {
if (!data) {
return;
}
const hideData = [];
const inUserLanguage = (d) => {
const dlocal = d.locale.split('-')[0] ?? d.locale;
if (language.cache.includes(dlocal)) {
return true;
}
hideData.push(d);
return false;
};
const filterLang = data.filter((d) => {
if (d.deleted) {
return false;
}
if (cfg.filterlang && !inUserLanguage(d)) {
return false;
}
return true;
});
let finalList = filterLang;
const hds = [];
for (const ujs of hideData) {
await reqCode(ujs);
if (ujs.translated) {
hds.push(ujs);
}
}
finalList = [...new Set([...hds, ...filterLang])];
for (const ujs of finalList) {
initUserJS(ujs, engine.name);
}
cache[engine.name].push(...finalList);
MUJS.updateCounter(finalList.length, engine);
};
const customFN = async (htmlDocument) => {
try {
if (!htmlDocument) {
return;
}
const selected = htmlDocument.documentElement;
if (/openuserjs/gi.test(engine.name)) {
for (const i of normalizeTarget(qsA('.col-sm-8 .tr-link', selected))) {
while (isNull(qs('.script-version', i))) {
await new Promise((resolve) => requestAnimationFrame(resolve));
}
const fixurl = dom
.prop(qs('.tr-link-a', i), 'href')
.replace(new RegExp(document.location.origin, 'gi'), 'https://openuserjs.org');
const ujs = mergeTemplate({
name: dom.text(qs('.tr-link-a', i)),
description: dom.text(qs('p', i)),
version: dom.text(qs('.script-version', i)),
url: fixurl,
code_url: `${fixurl.replace(/\/scripts/gi, '/install')}.user.js`,
total_installs: dom.text(qs('td:nth-child(2) p', i)),
created_at: dom.attr(qs('td:nth-child(4) time', i), 'datetime'),
code_updated_at: dom.attr(qs('td:nth-child(4) time', i), 'datetime'),
users: [
{
name: dom.text(qs('.inline-block a', i)),
url: dom.prop(qs('.inline-block a', i), 'href')
}
]
});
if (cfg.codePreview && !ujs.code_data) {
await reqCode(ujs);
}
createjs(ujs, engine.name);
customRecords.push(ujs);
}
}
cache[engine.name].push(...customRecords);
MUJS.updateCounter(customRecords.length, engine);
} catch (ex) {
MUJS.showError(ex);
}
};
const gitFN = async (data) => {
try {
if (isBlank(data.items)) return;
for (const r of data.items) {
const ujs = mergeTemplate({
name: r.name,
description: isEmpty(r.repository.description)
? i18n$('no_license')
: r.repository.description,
url: r.html_url,
code_url: r.html_url.replace(/\/blob\//g, '/raw/'),
code_updated_at: r.commit || Date.now(),
total_installs: r.score,
users: [
{
name: r.repository.owner.login,
url: r.repository.owner.html_url
}
]
});
if (cfg.codePreview && !ujs.code_data) {
await reqCode(ujs);
}
createjs(ujs, engine.name);
customRecords.push(ujs);
}
cache[engine.name].push(...customRecords);
MUJS.updateCounter(data.items.length, engine);
} catch (ex) {
MUJS.showError(ex);
}
};
if (engine.name.includes('fork')) {
Network.req(`${engine.url}/scripts/by-site/${host}.json`)
.then(forkFN)
.catch(MUJS.showError);
} else if (/github/gi.test(engine.name)) {
if (isEmpty(engine.token)) {
MUJS.showError(`"${engine.name}" requires a token to use`);
continue;
}
Network.req(
`${engine.url}"// ==UserScript=="+${host}+ "// ==/UserScript=="+in:file+language:js&per_page=30`,
'GET',
'json',
{
headers: {
Accept: 'application/vnd.github+json',
Authorization: `Bearer ${engine.token}`,
'X-GitHub-Api-Version': '2022-11-28'
}
}
)
.then(gitFN)
.then(() => {
Network.req('https://api.github.com/rate_limit', 'GET', 'json', {
headers: {
Accept: 'application/vnd.github+json',
Authorization: `Bearer ${engine.token}`,
'X-GitHub-Api-Version': '2022-11-28'
}
})
.then((data) => {
for (const [key, value] of Object.entries(data.resources.code_search)) {
const txt = make('mujs-row', 'rate-info', {
textContent: `${key.toUpperCase()}: ${value}`
});
rateContainer.append(txt);
}
})
.catch(MUJS.showError);
})
.catch(MUJS.showError);
} else {
Network.req(`${engine.url}${host}`, 'GET', 'document')
.then(customFN)
.catch((error) => {
MUJS.showError(`Engine: "${engine.name}"`, error);
});
}
}
urlBar.placeholder = i18n$('search_placeholder');
urlBar.value = '';
} catch (ex) {
MUJS.showError(ex);
}
};
//#endregion
//#region Make Config
const makecfg = () => {
const makerow = (desc, type = null, nm, attrs = {}) => {
desc = desc ?? i18n$('no_license');
nm = nm ?? i18n$('no_license');
const sec = make('mujs-section', 'mujs-cfg-section');
const lb = make('label');
const divDesc = make('mu-js', 'mujs-cfg-desc', {
textContent: desc
});
lb.append(divDesc);
sec.append(lb);
cfgpage.append(sec);
if (isNull(type)) {
return lb;
}
const inp = make('input', 'mujs-cfg-input', {
type,
dataset: {
name: nm
},
...attrs
});
if (!cfgMap.has(nm)) {
cfgMap.set(nm, inp);
}
if (type === 'checkbox') {
const inlab = make('mu-js', 'mujs-inlab');
const la = make('label', '', {
onclick() {
inp.dispatchEvent(new MouseEvent('click'));
}
});
inlab.append(inp, la);
lb.append(inlab);
if (nm.includes('-')) {
return inp;
}
if (/(greasy|sleazy)fork|openuserjs|gi(thub|st)/gi.test(nm)) {
for (const i of cfg.engines) {
if (i.name !== nm) continue;
inp.checked = i.enabled;
inp.dataset.engine = i.name;
ael(inp, 'change', (evt) => {
container.unsaved = true;
container.rebuild = true;
i.enabled = evt.target.checked;
});
}
} else {
inp.checked = cfg[nm];
ael(inp, 'change', (evt) => {
container.unsaved = true;
if (/filterlang/i.test(nm)) {
container.rebuild = true;
}
cfg[nm] = evt.target.checked;
});
}
} else {
lb.append(inp);
}
return inp;
};
if (isGM) {
makerow('Sync with GM', 'checkbox', 'cache');
}
makerow(i18n$('userjs_fullscreen'), 'checkbox', 'autoexpand', {
onchange(e) {
if (e.target.checked) {
dom.cl.add([btnfullscreen, main], 'expanded');
dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('fsClose'));
} else {
dom.cl.remove([btnfullscreen, main], 'expanded');
dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('fsOpen'));
}
}
});
makerow(i18n$('redirect'), 'checkbox', 'sleazyredirect');
makerow(i18n$('filter'), 'checkbox', 'filterlang');
makerow(i18n$('preview_code'), 'checkbox', 'codePreview');
for (const inp of [
makerow('Recommend author', 'checkbox', 'recommend-author'),
makerow('Recommend scripts', 'checkbox', 'recommend-others')
]) {
const nm = inp.dataset.name === 'recommend-author' ? 'author' : 'others';
inp.checked = cfg.recommend[nm];
ael(inp, 'change', (evt) => {
container.unsaved = true;
cfg.recommend[nm] = evt.target.checked;
});
}
makerow('Greasy Fork', 'checkbox', 'greasyfork');
makerow('Sleazy Fork', 'checkbox', 'sleazyfork');
makerow('Open UserJS', 'checkbox', 'openuserjs');
makerow('GitHub API', 'checkbox', 'github');
const ghAPI = cfg.engines.find((c) => c.name === 'github');
const ghToken = makerow('GitHub API (Token)', 'text', 'github', {
defaultValue: '',
value: ghAPI.token ?? '',
placeholder: 'Paste Access Token',
onchange(evt) {
container.unsaved = true;
container.rebuild = true;
if (isNull(ghMsg)) {
ghAPI.token = evt.target.value;
}
}
});
ghToken.dataset.engine = 'github-token';
cfgMap.set('github-token', ghToken);
makerow(`${i18n$('dtime')} (ms)`, 'number', 'time', {
defaultValue: 10000,
value: cfg.time,
min: 0,
step: 500,
onbeforeinput(evt) {
if (evt.target.validity.badInput) {
dom.cl.add(evt.target, 'mujs-invalid');
dom.prop(savebtn, 'disabled', true);
} else {
dom.cl.remove(evt.target, 'mujs-invalid');
dom.prop(savebtn, 'disabled', false);
}
},
oninput(evt) {
container.unsaved = true;
const t = evt.target;
if (t.validity.badInput || (t.validity.rangeUnderflow && t.value !== '-1')) {
dom.cl.add(t, 'mujs-invalid');
dom.prop(savebtn, 'disabled', true);
} else {
dom.cl.remove(t, 'mujs-invalid');
dom.prop(savebtn, 'disabled', false);
cfg.time = isEmpty(t.value) ? cfg.time : parseFloat(t.value);
}
}
});
const cbtn = make('mu-js', 'mujs-sty-flex');
const savebtn = make('mujs-btn', 'save', {
textContent: i18n$('save'),
dataset: {
command: 'save'
},
disabled: false
});
const resetbtn = make('mujs-btn', 'reset', {
textContent: i18n$('reset'),
dataset: {
command: 'reset'
}
});
const blacklist = make('textarea', '', {
dataset: {
name: 'blacklist'
},
rows: '10',
autocomplete: false,
spellcheck: false,
wrap: 'soft',
value: JSON.stringify(cfg.blacklist, null, ' '),
oninput(evt) {
let isvalid = true;
try {
cfg.blacklist = JSON.parse(evt.target.value);
isvalid = true;
} catch (ex) {
err(ex);
isvalid = false;
} finally {
if (isvalid) {
dom.cl.remove(evt.target, 'mujs-invalid');
dom.prop(savebtn, 'disabled', false);
} else {
dom.cl.add(evt.target, 'mujs-invalid');
dom.prop(savebtn, 'disabled', true);
}
}
}
});
const theme = make('textarea', '', {
dataset: {
name: 'theme'
},
rows: '10',
autocomplete: false,
spellcheck: false,
wrap: 'soft',
value: JSON.stringify(cfg.theme, null, ' '),
oninput(evt) {
let isvalid = true;
try {
cfg.theme = JSON.parse(evt.target.value);
container.renderTheme(JSON.parse(evt.target.value));
isvalid = true;
} catch (ex) {
err(ex);
isvalid = false;
} finally {
if (isvalid) {
dom.cl.remove(evt.target, 'mujs-invalid');
dom.prop(savebtn, 'disabled', false);
} else {
dom.cl.add(evt.target, 'mujs-invalid');
dom.prop(savebtn, 'disabled', true);
}
}
}
});
cfgMap.set('blacklist', blacklist);
cfgMap.set('theme', theme);
cbtn.append(resetbtn, savebtn);
cfgpage.append(blacklist, theme, cbtn);
};
//#endregion
const makeTHead = (rows) => {
const tr = make('tr');
for (const r of normalizeTarget(rows)) {
const tparent = make('th', r.class ?? '', r);
tr.append(tparent);
}
tabhead.append(tr);
table.append(tabhead, tabbody);
};
ael(mainframe, 'mouseenter', (evt) => {
evt.preventDefault();
evt.stopPropagation();
evt.target.style.opacity = container.opacityMax;
frameTimeout.clear(...frameTimeout.ids);
});
ael(mainframe, 'mouseleave', (evt) => {
evt.preventDefault();
evt.stopPropagation();
evt.target.style.opacity = container.opacityMin;
timeoutFrame();
});
ael(mainframe, 'click', (evt) => {
evt.preventDefault();
frameTimeout.clear(...frameTimeout.ids);
dom.cl.remove(main, 'hidden');
dom.cl.add(mainframe, 'hidden');
if (cfg.autoexpand) {
dom.cl.add([btnfullscreen, main], 'expanded');
dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('fsClose'));
}
if (dom.cl.has(mainframe, 'error')) {
tab.create('mujs:settings');
}
});
ael(urlBar, 'input', (evt) => {
evt.preventDefault();
if (urlBar.placeholder === i18n$('newTab')) {
return;
}
const val = evt.target.value;
if (isEmpty(val)) {
dom.cl.remove(qsA('tr[data-engine]', tabbody), 'hidden');
return;
}
const reg = new RegExp(val, 'gi');
const finds = new Set();
const userjsCache = container.userjsCache;
for (const [k, v] of userjsCache) {
const elem = qs(`tr[data-script-id="${k}"]`, tabbody);
if (!elem) {
continue;
}
if (finds.has(elem)) {
continue;
}
if (v.name && v.name.match(reg)) {
finds.add(elem);
}
if (v.description && v.description.match(reg)) {
finds.add(elem);
}
if (v.code_data) {
const meta = parse_meta(v.code_data);
for (const key of Object.keys(meta)) {
if (/name|desc/i.test(key) && key.match(reg)) {
finds.add(elem);
}
}
}
}
dom.cl.add(qsA('tr[data-engine]', tabbody), 'hidden');
dom.cl.remove([...finds], 'hidden');
});
ael(urlBar, 'change', (evt) => {
evt.preventDefault();
const val = evt.target.value;
if (urlBar.placeholder === i18n$('newTab') && qs('mujs-tab.active', toolbar)) {
const tabElem = qs('mujs-tab.active', toolbar);
const tabHost = qs('mujs-host', tabElem);
if (val.startsWith('mujs:')) {
tab.close(tabElem);
if (tab.hasTab(val)) {
tab.active(tab.Tab.get(val));
} else {
tab.create(val);
}
return;
} else if (val === '*') {
tabElem.dataset.host = val;
tabHost.title = '';
tabHost.textContent = '';
buildlist(val);
return;
}
const value = container.getHost(val);
if (MUJS.checkBlacklist(value)) {
MUJS.showError(`Host blacklisted "${value}"`);
return;
}
tabElem.dataset.host = value;
tabHost.title = value;
tabHost.textContent = value;
buildlist(value);
return;
}
});
container.renderTheme(cfg.theme);
makeTHead([
{
class: 'mujs-header-name',
textContent: i18n$('name')
},
{
textContent: i18n$('createdby')
},
{
textContent: i18n$('daily_installs')
},
{
textContent: i18n$('updated')
},
{
textContent: i18n$('install')
}
]);
const getCellValue = (tr, idx) => tr.children[idx].innerText || tr.children[idx].textContent;
const comparer = (idx, asc) => (a, b) =>
((v1, v2) =>
v1 !== '' && v2 !== '' && !isNaN(v1) && !isNaN(v2)
? v1 - v2
: v1.toString().localeCompare(v2))(
getCellValue(asc ? a : b, idx),
getCellValue(asc ? b : a, idx)
);
for (const th of tabhead.rows[0].cells) {
if (dom.text(th) === i18n$('install')) continue;
dom.cl.add(th, 'mujs-pointer');
ael(th, 'click', () => {
/** [Stack Overflow Reference](https://stackoverflow.com/questions/14267781/sorting-html-table-with-javascript/53880407#53880407) */
const table = th.closest('table');
const tbody = table.querySelector('tbody');
Array.from(tbody.querySelectorAll('tr'))
.sort(comparer(Array.from(th.parentNode.children).indexOf(th), (this.asc = !this.asc)))
.forEach((tr) => tbody.appendChild(tr));
});
}
makecfg();
tab.create(container.host);
buildlist().then(timeoutFrame);
dbg('Container', container);
} catch (ex) {
err(ex);
container.remove();
}
}
// #endregion
/**
* @template { Function } F
* @param { (this: F, doc: Document) => any } onDomReady
*/
const loadDOM = (onDomReady) => {
if (!isFN(onDomReady)) {
return;
}
if (document.readyState === 'interactive' || document.readyState === 'complete') {
onDomReady.call({}, document);
}
document.addEventListener('DOMContentLoaded', (evt) => onDomReady.call({}, evt.target), {
once: true
});
};
const updateList = () => {
try {
const oldList = normalizeTarget(cfg.blacklist).filter(
(b) => typeof b?.name === 'string' && /Blacklist/.test(b.name)
);
if (isBlank(oldList)) {
return;
}
const legacy = [
{
enabled: true,
regex: true,
flags: '',
name: 'Blacklist 1',
url: '(gov|cart|checkout|login|join|signin|signup|sign-up|password|reset|password_reset)'
},
{
enabled: true,
regex: true,
flags: '',
name: 'Blacklist 2',
url: '(pay|bank|money|localhost|authorize|checkout|bill|wallet|router)'
},
{
enabled: true,
regex: false,
flags: '',
name: 'Blacklist 3',
url: 'https://home.bluesnap.com'
},
{
enabled: true,
regex: false,
flags: '',
name: 'Blacklist 4',
url: ['zalo.me', 'skrill.com']
}
];
let toUpdate = false;
for (let i = 0; i < oldList.length; i++) {
const o = oldList[i];
const p = legacy[i];
if (o.url === p.url) {
continue;
}
toUpdate = true;
}
if (toUpdate) {
const list = normalizeTarget(cfg.blacklist);
if (oldList.length === list.length) {
cfg.blacklist = defcfg.blacklist;
container.save();
return;
}
for (let i = 0; i < list.length; i++) {
const b = list[i];
if (typeof b?.name === 'string' && /Blacklist/.test(b.name)) {
delete list[i];
list[i] = defcfg.blacklist[i];
}
}
container.save();
return;
}
container.oldBlacklist = true;
} catch (ex) {
err(ex);
}
};
const init = async () => {
const stored = await StorageSystem.getValue('Config', defcfg);
cfg = setObj(defcfg, stored);
updateList();
info('Config:', cfg);
loadDOM((doc) => {
try {
if (window.location === null) {
err('"window.location" is null, reload the webpage or use a different one');
return;
}
if (doc === null) {
err('"doc" is null, reload the webpage or use a different one');
return;
}
sleazyRedirect();
container.inject(primaryFN, doc);
} catch (ex) {
err(ex);
}
});
};
init();
})();