// ==UserScript== // @version 6.7.3 // @name Magic Userscript+ : Show Site All UserJS // @name:ar Magic Userscript+: عرض جميع ملفات UserJS // @name:de Magic Userscript+ : Website anzeigen Alle 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:pl Magic Userscript+ : Pokaż witrynę Wszystkie UserJS // @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:de Findet verfügbare Benutzerskripte für die aktuelle Webseite. // @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:pl Wyszukuje dostępne skrypty użytkownika dla bieżącej strony internetowej. // @description:ru Находит доступные юзерскрипты для текущей веб-страницы. // @description:zh 为当前网页查找可用的用户脚本。 // @description:zh-CN 为当前网页查找可用的用户脚本。 // @description:zh-TW 为当前网页查找可用的用户脚本。 // @supportURL https://github.com/magicoflolis/Userscript-Plus/issues // @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_registerMenuCommand // @grant GM_xmlhttpRequest // @grant GM.addElement // @grant GM.info // @grant GM.getValue // @grant GM.openInTab // @grant GM.setValue // @grant GM.registerMenuCommand // @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":{"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":"استيراد النسق","code_size":"حجم الرمز"},"de":{"createdby":"Erstellt von","name":"Name","daily_installs":"Tägliche Installationen","close":"Schließen Sie","filterA":"Filter","max":"Maximieren Sie","min":"minimieren","search":"Suche","search_placeholder":"Suche nach Userscripts","install":"Installieren Sie","issue":"Neue Ausgabe","version_number":"Version","updated":"Zuletzt aktualisiert","total_installs":"Installationen insgesamt","ratings":"Bewertungen","good":"Gut","ok":"Okay","bad":"Schlecht","created_date":"Erstellt","redirect":"Greasy Fork für Erwachsene","filter":"Andere Sprachen herausfiltern","dtime":"Zeitüberschreitung anzeigen","save":"Speichern Sie","reset":"Zurücksetzen","preview_code":"Vorschau Code","saveFile":"Datei speichern","newTab":"Neue Registerkarte","applies_to":"Gilt für","license":"Lizenz","no_license":"N/A","antifeatures":"Antifeatures","userjs_fullscreen":"Automatischer Vollbildmodus","listing_none":"(Keine)","export_config":"Konfig exportieren","export_theme":"Thema exportieren","import_config":"Konfig importieren","import_theme":"Thema importieren","code_size":"Code Größe"},"en":{"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":"Okay","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","code_size":"Code Size"},"en_GB":{"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","code_size":"Code Size"},"es":{"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","code_size":"Código Tamaño"},"fr":{"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","code_size":"Code Taille"},"ja":{"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":"テーマのインポート","code_size":"コード・サイズ"},"nl":{"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","code_size":"Code Grootte"},"pl":{"createdby":"Stworzony przez","name":"Nazwa","daily_installs":"Codzienne instalacje","close":"Zamknij","filterA":"Filtr","max":"Maksymalizuj","min":"Minimalizuj","search":"Wyszukiwanie","search_placeholder":"Wyszukiwanie skryptów użytkownika","install":"Instalacja","issue":"Nowy numer","version_number":"Wersja","updated":"Ostatnia aktualizacja","total_installs":"Łączna liczba instalacji","ratings":"Oceny","good":"Dobry","ok":"Ok","bad":"Zły","created_date":"Utworzony","redirect":"Greasy Fork dla dorosłych","filter":"Odfiltruj inne języki","dtime":"Limit czasu wyświetlania","save":"Zapisz","reset":"Reset","preview_code":"Kod podglądu","saveFile":"Zapisz plik","newTab":"Nowa karta","applies_to":"Dotyczy","license":"Licencja","no_license":"N/A","antifeatures":"Antywzorce","userjs_fullscreen":"Automatyczny pełny ekran","listing_none":"(Brak)","export_config":"Konfiguracja eksportu","export_theme":"Motyw eksportu","import_config":"Importuj konfigurację","import_theme":"Importuj motyw","code_size":"Kod Rozmiar"},"ru":{"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":"Импортировать тему","code_size":"Код Размер"},"zh":{"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":"导入主题","code_size":"代码 尺寸"},"zh_CN":{"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":"导入主题","code_size":"代码 尺寸"},"zh_TW":{"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":"导入主题","code_size":"代码 尺寸"}}; /** * 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}mujs-main .mujs-prompt{align-items:center;justify-content:center}mujs-main .mujs-prompt svg{width:14px;height:14px;background:rgba(0,0,0,0)}mujs-main .mujs-prompt>.prompt{position:absolute;background:var(--mujs-background-color, hsl(222, 14%, 33%)) !important;border:1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));border-radius:16px;text-align:center;padding:.5em;z-index:1}mujs-main .mujs-prompt>.prompt .prompt-head{font-size:18px}mujs-main .mujs-prompt>.prompt .prompt-body{display:grid;grid-auto-flow:column;grid-gap:.5em;padding-top:.5em}mujs-main .mujs-prompt>.prompt mujs-btn[data-command=prompt-deny]{background:var(--mujs-sf-btn-color, hsl(12, 86%, 50%));border-color:var(--mujs-sf-btn-color, hsl(12, 86%, 50%))}mujs-main .mujs-prompt>.prompt mujs-btn[data-command=prompt-deny]:hover{background:var(--mujs-sf-txt-color, hsl(12, 79%, 55%));border-color:var(--mujs-sf-txt-color, hsl(12, 79%, 55%))}mujs-main .mujs-prompt>.prompt mujs-btn[data-command=prompt-confirm]{background:var(--mujs-gf-color, hsl(204, 100%, 40%));border-color:var(--mujs-gf-color, hsl(204, 100%, 40%))}mujs-main .mujs-prompt>.prompt mujs-btn[data-command=prompt-confirm]:hover{background:var(--mujs-gf-btn-color, hsl(211, 87%, 56%));border-color:var(--mujs-gf-btn-color, hsl(211, 87%, 56%))}.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%}}@media screen and (max-width: 550px){mujs-body table .frame:not(.webext-page) td{margin:1rem !important}mujs-body table .frame:not(.webext-page) td:not(.mujs-name,.install-btn){width:auto !important}}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} `; const inIframe = () => { try { return window.self !== window.top; } catch (e) { return true; } } if (inIframe()) { return; } 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; } /******************************************************************************/ // TODO: add translations to "Install as user style?" // #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; /** @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, trustedTypes: g.trustedTypes }; 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 winLocation = window.top.location; 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] ?? translations['en']; 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 = (obj, prop) => { if (typeof Object.hasOwn !== 'undefined') { return Object.hasOwn(obj, prop); } return Object.prototype.hasOwnProperty.call(obj, prop); }; /** * @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 { str = str ?? window.location ?? 'about:blank'; return objToStr(str).includes('URL') ? str : new URL(str); } 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").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, options); 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, cname); } } 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 (isObj(GM.info)) { return GM.info; } else if (isObj(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/Userscript-Plus/issues' } }; }; const $info = getGMInfo(); // #endregion /** * @type { import("../typings/UserJS.d.ts").dom } */ const dom = { 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); } } }, prop(target, prop, value = undefined) { for (const elem of normalizeTarget(target)) { if (value === undefined) { return elem[prop]; } elem[prop] = value; } }, 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; } }, cl: { add(target, token) { if (Array.isArray(token)) { for (const elem of normalizeTarget(target)) { elem.classList.add(...token); } } else { for (const elem of normalizeTarget(target)) { elem.classList.add(token); } } }, remove(target, token) { if (Array.isArray(token)) { for (const elem of normalizeTarget(target)) { elem.classList.remove(...token); } } else { for (const elem of normalizeTarget(target)) { elem.classList.remove(token); } } }, toggle(target, token, force) { let r; for (const elem of normalizeTarget(target)) { r = elem.classList.toggle(token, force); } return r; }, has(target, token) { for (const elem of normalizeTarget(target)) { if (elem.classList.contains(token)) { return true; } } return false; } } }; class Memorize { constructor() { /** * @type {Map>} */ 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(); //#region Icon SVGs 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: '' }, refresh: { viewBox: '0 0 1024 1024', fill: '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); } try { if (typeof iconSVG[type].html === 'string') { svgElem.innerHTML = iconSVG[type].html; } // eslint-disable-next-line no-unused-vars } catch (ex) { /* empty */ } if (container) { container.appendChild(svgElem); return svgElem; } return svgElem.outerHTML; } }; //#endregion /** * @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); } } }; class jsCommand { constructor() { this.cmds = []; } register(text, command) { if (!isGM) { return; } if (isFN(command)) { const found = this.cmds.find(c => command === c); if (found) { return; } this.cmds.push(command); } if (isFN(GM.registerMenuCommand)) { GM.registerMenuCommand(text, command); } else if (isFN(GM_registerMenuCommand)) { GM_registerMenuCommand(text, command); } } }; const Command = new jsCommand(); /** * @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.log(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' ); info(`Redirecting to "${str}"`); 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); 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: $info.script.name, role: 'primary-container' } }) : make('iframe', 'mujs-iframe', { dataset: { insertedBy: $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 "${this.host}"`); 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 = $info.script.name; sty.dataset.role = name; return sty; } } const sty = make('style', '', { textContent: css, dataset: { insertedBy: $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; if (/accounts*\.google\./.test(this.webpage.host)) { blacklisted = true; } 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); 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 promptElem = make('mujs-row', 'mujs-prompt'); 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${ $info.script.version.includes('.') || $info.script.version.includes('Book') ? $info.script.version : $info.script.version.slice(0, 5) })`, innerHTML: iconSVG.load('gh'), dataset: { command: 'open-tab', webpage: $info.script.namespace } }); const btnissue = make('mujs-btn', 'issue hidden', { innerHTML: iconSVG.load('issue'), title: i18n$('issue'), dataset: { command: 'open-tab', webpage: $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, promptElem); 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 prmpt = /prompt-/.test(target.dataset.command); let dataset = target.dataset; let cmd = dataset.command; let prmptChoice = false; if (prmpt) { dataset = target.parentElement.dataset; cmd = dataset.command; prmptChoice = /confirm/.test(target.dataset.command); target.parentElement.parentElement.remove(); } if (cmd === 'install-script' && dataset.userjs) { let installCode = dataset.userjs; if (!prmpt && dataset.userjs.endsWith('.user.css')) { MUJS.makePrompt('Install as user style?', dataset); return; } else if (prmpt !== prmptChoice) { installCode = dataset.userjs.replace(/\.user\.css$/, '.user.js'); }; const dlBtn = make('a', '', { onclick(evt) { evt.preventDefault(); doInstallProcess(evt.target); } }); dlBtn.href = installCode; 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 qsA('.error', footer)) { elem.remove(); } } 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 "${str}"`); 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); } makePrompt(txt, dataset = {}, usePrompt = true) { if (qs('.prompt', promptElem)) { for (const elem of qsA('.prompt', promptElem)) { if (elem.dataset.prompt) { elem.remove(); } } } const el = make('mu-js', 'prompt', { dataset: { prompt: txt } }); const elHead = make('mu-js', 'prompt-head', { innerHTML: `${iconSVG.load('refresh')} ${txt}` }); el.append(elHead); if (usePrompt) { const elPrompt = make('mu-js', 'prompt-body', { dataset }); const elYes = make('mujs-btn', 'prompt-confirm', { innerHTML: 'Confirm', dataset: { command: 'prompt-confirm' } }); const elNo = make('mujs-btn', 'prompt-deny', { innerHTML: 'Deny', dataset: { command: 'prompt-deny' } }); elPrompt.append(elYes, elNo); el.append(elPrompt); } promptElem.append(el); } 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_meta: {}, 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); if (translate) { for (const lng of language.cache) { if (meta[`name:${lng}`]) { Object.assign(obj, { name: meta[`name:${lng}`], translated: true }); } if (meta[`description:${lng}`]) { Object.assign(obj, { description: meta[`description:${lng}`], translated: true }); } } } 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_meta: meta, code_match: applies_to_names, code_grant: [...grantSet], antifeatures: [...afSet] }); 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')) { // fdaily.dataset.command = 'open-tab'; // fdaily.dataset.webpage = `${ujs.url}/stats`; // fupdated.dataset.command = 'open-tab'; // fupdated.dataset.webpage = `${ujs.url}/versions`; // } 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(i18n$('code_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 (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); }; const arr = []; 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; } /** * @param { import("../typings/UserJS.d.ts").GSFork } dataQ */ const forkFN = async (dataQ) => { if (!dataQ) { MUJS.showError('Invalid data received from the server, check internet connection'); return; } const dq = Array.isArray(dataQ) ? dataQ : Array.isArray(dataQ.query) ? dataQ.query : []; const data = dq.filter((d) => !d.deleted); if (isBlank(data)) { return; } const hideData = []; /** * @param {import("../typings/UserJS.d.ts").GSForkQuery} d * @returns {boolean} */ 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 (cfg.filterlang && !inUserLanguage(d)) { return false; } return true; }); let finalList = filterLang; const hds = []; for (const ujs of hideData) { await reqCode(ujs, true); 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); }; /** * @param {Document} htmlDocument */ const customFN = async (htmlDocument) => { try { if (!htmlDocument) { MUJS.showError('Invalid data received from the server, TODO fix this'); return; } const selected = htmlDocument.documentElement; if (/openuserjs/gi.test(engine.name)) { for (const i of 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)) { MUJS.showError('Invalid data received from the server, TODO fix this'); 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); } }; let netFN; if (engine.name.includes('fork')) { netFN = 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; } netFN = 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 { netFN = Network.req(`${engine.url}${host}`, 'GET', 'document') .then(customFN) .catch((error) => { MUJS.showError(`Engine: "${engine.name}"`, error); }); } if (netFN) { arr.push(netFN); } } urlBar.placeholder = i18n$('search_placeholder'); urlBar.value = ''; Promise.allSettled(arr).then(() => { tabhead.rows[0].cells[2].dispatchEvent(new MouseEvent('click')); tabhead.rows[0].cells[2].dispatchEvent(new MouseEvent('click')); }); } 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 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); 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); }; 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) */ Array.from(tabbody.querySelectorAll('tr')) .sort(comparer(Array.from(th.parentNode.children).indexOf(th), (this.asc = !this.asc))) .forEach((tr) => tabbody.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 init = async () => { const stored = await StorageSystem.getValue('Config', defcfg); cfg = { ...defcfg, ...stored }; info('Config:', cfg); loadDOM((doc) => { try { if (window.location === null) { throw new Error('"window.location" is null, reload the webpage or use a different one'); } if (doc === null) { throw new Error('"doc" is null, reload the webpage or use a different one'); } if (window.trustedTypes && window.trustedTypes.createPolicy) { window.trustedTypes.createPolicy('default', { createHTML: (string) => string }); } sleazyRedirect(); container.inject(primaryFN, doc); Command.register('Inject Userscript+', () => { container.inject(primaryFN, doc); }); Command.register('Close Userscript+', () => { container.remove(); }); } catch (ex) { err(ex); } }); }; init(); })();