🏠 返回首頁 

Greasy Fork is available in English.

Magic Userscript+: показать сайт всем UserJS

Находит доступные юзерскрипты для текущей веб-страницы.


Установить этот скрипт?
  1. // ==UserScript==
  2. // @version 7.6.4
  3. // @name Magic Userscript+ : Show Site All UserJS
  4. // @name:ar Magic Userscript+: عرض جميع ملفات UserJS
  5. // @name:de Magic Userscript+ : Website anzeigen Alle UserJS
  6. // @name:es Magic Userscript+: Mostrar sitio todos los UserJS
  7. // @name:fr Magic Userscript+ : Afficher le site Tous les UserJS
  8. // @name:ja Magic Userscript+ : サイトをすべて表示 UserJS
  9. // @name:nl Magic Userscript+: Site alle UserJS tonen
  10. // @name:pl Magic Userscript+ : Pokaż witrynę Wszystkie UserJS
  11. // @name:ru Magic Userscript+: показать сайт всем UserJS
  12. // @name:zh Magic Userscript+ :显示站点所有 UserJS
  13. // @name:zh-CN Magic Userscript+ :显示站点所有 UserJS
  14. // @name:zh-TW Magic Userscript+ :显示站点所有 UserJS
  15. // @description Finds available userscripts for the current webpage.
  16. // @description:ar يبحث عن نصوص المستخدمين المتاحة لصفحة الويب الحالية.
  17. // @description:de Findet verfügbare Benutzerskripte für die aktuelle Webseite.
  18. // @description:es Busca los usercripts disponibles para la página web actual.
  19. // @description:fr Recherche les userscripts disponibles pour la page web en cours.
  20. // @description:ja 現在のウェブページで利用可能なユーザスクリプトを検索します。
  21. // @description:nl Zoekt beschikbare gebruikerscripts voor de huidige webpagina.
  22. // @description:pl Wyszukuje dostępne skrypty użytkownika dla bieżącej strony internetowej.
  23. // @description:ru Находит доступные юзерскрипты для текущей веб-страницы.
  24. // @description:zh 为当前网页查找可用的用户脚本。
  25. // @description:zh-CN 为当前网页查找可用的用户脚本。
  26. // @description:zh-TW 为当前网页查找可用的用户脚本。
  27. // @author Magic <magicoflolis@tuta.io>
  28. // @supportURL https://github.com/magicoflolis/Userscript-Plus/issues
  29. // @namespace https://github.com/magicoflolis/Userscript-Plus
  30. // @homepageURL https://github.com/magicoflolis/Userscript-Plus
  31. // @icon data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEAAAABACAYAAACqaXHeAAABhGlDQ1BJQ0MgcHJvZmlsZQAAKJF9kT1Iw0AcxV8TRZFKBzuIOGSoTnZREcGlVLEIFkpboVUHk0u/oElDkuLiKLgWHPxYrDq4OOvq4CoIgh8gzg5Oii5S4v+SQosYD4778e7e4+4dIDSrTLN6YoCm22Y6EZdy+VWp7xUigghhDqLMLCOZWczCd3zdI8DXuyjP8j/35xhUCxYDAhJxjBmmTbxBPLNpG5z3icOsLKvE58QTJl2Q+JHrisdvn###CzwzbGbT88RhYqnUxUoXs7KpEU8TR1RNp3wh57HKeYuzVq2z9j35C4MFfSXDdZqjSGAJSaQgQUEdFVRhI0qrToqFNO3Hffwjrj9FLoVcFTByLKAGDbLrB/+D391axalJLykYB3pfHOdjDOjbBVoNx/k+dpzWCSA+A1d6x19rArOfpDc6WuQICG0DF9cdTdkDLneA4SdDNmVXEmkKxSLwfkbflAeGboGBNa+39j5OH4AsdbV8AxwcAuMlyl73eXd/d2//nmn39wOjunK6jS33SAAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB+gDDBAAJWyXgRAAABRPSURBVHjazVt9VNRl9v9854VhhjdBiHcUaFQEDTVCdk1IstTV0jb1LLZkJZrl4knLTnZaKjdqdXdLXcvfyoHUErPUBBEMLREJ6egJFBUUBF9WMd6R4WW+M/P5/eHMdxkYEJS0e8498MzzzPN87537PM+9n3u/An49kgFwA+AJIACAP4AxAEIA+AFwBzDEPLYJQB2AqwDOATgN4AqAywBubNu2rTk+Pl4EAJIyABAEwTQYDyn8CoIrAEQD+BOAMLPgLgAcAGDs2LGQyWSora1FYGAg6urqUFZWBgCIiYnBqVOn0NDQAAA6T0/Pm/X19ZflcvkZo9G4w2AwHAFgwG+Q5ACCg4KClixZsuT82rVrTTt27CCAHpybm8vq6moWFRXRZDKxpqaGkyZN4q0fl3zyySelsaIoMiYmhg899BD/9re/mQAUAogHEGxe8/5TaGiob0RERMrSpUuvZ2Zmmo4cOcIjR44wLy/PpgIOHjzI2tpaTp06lVqtlhUVFUxLS5MU8MQTT0hjLQqZPHkySRIAFy1aZAwNDa0C8C/z9rpvNBbAPwBctyVob3zw4EEWFBRQJpNJFvH999/3WwE5OTncs2ePZUwVgDjzWXPHB9Wd0GsACuRy+UoXFxev5cuXo7OzEyStWBBsHzGiKMJkunWGmUwmyGQy6X+lUgkAcHBw6HXxoUOHWv4dDiAFwNcAJtzpgTUQUgNYIgjCP2bNmiX7+9//jo0bN2Lz5s3IyspCUFCQJIz5xB7Q5FVVVZg9ezby8vKwYsWKgTzTFACZAFYC+BZA+6ArQKlUOouiuAZAgiAIMq1WixEjRuCjjz6Cq6srPvzwQ1RUVNze5GQyKyV1bW/atAnJycmIj49Henr6/66qLpbUdbzRaMS0adOQm5sLAN4A/g/ARADvAGgZvCNeLvdUq9VbzVcQAXDo0KHcs2cPDQYD29ramJiYSKVSedszICIigpGRkTbb9vb2nDRpEmNiYujk5MQZM2bQ3d2dbm5unDFjhtX4qKgonj59mt7e3t3XEAH82+xnDAp5AThjSxgPDw9++eWXNJlMbGpq4htvvCEdbr82r1ixgr/73e966zeaz4W7VoIzgAyZTMaoqChOnTqVarW6hxIyMzNpMBio0+m4ZMmSflnCPWCj2RKc71R4NYD1AIwODg4sKipiU1MTV69e3UNAT09Pfv311yTJxsZGLl++/J5Zwm1YNMugvhMFxAFoA0CFQsH33nuPHR0dbGtr49KlS6lSqawW8/LyYnZ2Ng0GA2/evMkXXniBCoXit6CEFrNbPiB6SalUtncVwNnZmcnJySTJlpYWvv7665TL5VaL+fj4cN++fSTJ+vp6vvzyy31agrOzc1/7eMAcFRXFIUOG2Oq7NhA/YaRcLr85bNgwuri4WE2kUqn48ccfs7OzkzqdjgkJCT0swdfXl7m5uTQajWxubuaCBQskRYWGhjIrK4uCIBAAY2JiSFJq3y2T5PTp03vrP9wfj1EA8PkjjzzC8+fPMysriz4+PlYTubi4cN26dSTJ5uZmm/vdz8+PBw4cIEnW1tbyxRdfpCAIDA4OpsFgsBo7WML3Y64287buk7x9fX07oqOj2dLSQpPJxMzMzB5KUKvV3LRpE/V6PW/evMmFCxfSzs7Oaoy/vz9/+OEHGo1GNjY2cu7cudRoNExNTeWePXuYkZHBsLAw7t+/n4IgUBAEPv/889y+fTu/+OILvvDCC5JAGRkZjIiI4OrVq/nll19aRYxdOSMjg+PHjycAJiUl8cMPP+TMmTO5Y8cOzpo1yxI7ePYV0v5rzpw51Gg0/Mtf/kKdTkeDwcCcnBx6enpaLebq6sr169dLJ7+t/a7Vann+/HmS5I0bN7hgwQK6urpy1apVLCwslLaASqViSkoKW1tbWVJSwuLiYt68eZNpaWlUKpUkyR9//JE///wzL126xOrqagYFBdncAhblZGdns66ujoWFhTx16hQ7Ozvp7+9PcxRpM5QOBlBu2cdOTk5877332NraSpPJxN27d9PLy8tqQQcHB27ZsoWiKLKlpYULFiyQrkhHR0e+++671Ov1tFB9fT1nz54tKSo6OpokOWbMGDY0NPC1116jWq2mvb09ly9fzsbGRoaHh5MkP/nkE9rb23PMmDESTnA7BVy4cIHBwcF0cXHhTz/9xH/+8580yxhsSwHx5nuT0dHRTEpKokql4sqVK9ne3k5RFLl//356eHhYLerm5sbPPvtMEvDFF1+kvb09165dy/b2dnan69ev86mnnqIgCJJwlnC366Hr7OxMk8nE2NhYkpRcYVvASW8KyM/Pl/qys7OZk5Nj8Q3ibQVFhZbBjz32GHU6HQFQqVQyOTmZOp2ORqOR6enpPZTg5OTErVu30mAwsKmpiSkpKezs7GRvdO7cOXp7e9PHx4cGg4GRkZHs7Ozkww8/LM05fvx4dnR0cNKkST0E7q8Cjh49aqUAC0gjCEJh90Aw1uw6SgqwABAWc37rrbfY0dFBURS5d+9eurm5WS3u7u7OlJQU9of0ej2fffZZOjk5UafT0dPTkydOnGBpaSmfeeYZzpkzh6dOnWJxcTG9vLwGTQGWtqOjo9EsM2Rm/lNXcKQ7kOHs7IwNGzZg48aNEEURTz31FD7++OOuwATa2tpw/vx5GAy3xyyVSiV8fX1hMplgMplw48YNvPTSS7h48SJ27NiB9PR0XL58GQsXLkRNTY3t+7oXsMXyeV9h9/z5861kdgdwvPt92tULtFxTarWaSUlJ1Ov11Ov13LVrF4cMGUKlUsl33nmHOp2uXxYgiiLnz59Pi5ttWUcmk1GhUFChUFjdKAqFwuqO79629blcLrfyVLu2zX+PW6LFUAD/7a+zoVKp+Mknn7C9vZ0Gg4Fbtmzh66+/zo6ODvaXzp07R19f3/sdI/zXLDumA2gdyJcdHByYnJxMURSp1+v7PPC6k8lk4ubNm38LIXMrgOlyM572zEAiJVEUUVBQAHd3d4wbNw4qlar/mRhBwKhRo3DlyhWcPXtWAkfvA9kBOCoHsBDAIwP9tsFgwNGjR6HRaBAZGWl14Nx2ZTs7PProo6irq0NxcfGAwdNBpEsAcOhOzSgoKIhqtZr/+c9/BnQGWKilpYVxcXE2cYO3336b77777q+9DQ4BQNmdfDk0NJS5ubm0s7Ojs7Mz169fT6PRaHPP90W//PIL4+Pje8QRmzZtYkpKyqAIGh0dzRMnTrC5uZm7du2in5+fpa8M5qzsgCaUyWRMT0+3+NbS7fDBBx+wvLyc9fX1rK2t5cmTJ/nSSy/xq6++oiiKvSqhqamJc+fO/VUQpMDAQNbW1jI9PZ0JCQksLy/nd999R3t7ewKow7fffmuqq6tjWlqapBlRFBkfH8+srCzJJe4e6lZUVEggaVlZGbVaLRUKBYOCgjhp0iRGRkbS29ubgiDQ3d2d27dv79MSampqOH/+fOke379/Pw8fPmxz24WHh9vkBx98sMf41atXs7KykgEBAQTAWbNmsba2lsHBwQRgwLx584wrVqxgdXU18/PzOXToUJJkVVUVk5KSGBcX12PShx9+WHJhAXDt2rVWgYeDgwMfe+wxenh48NVXX6W/vz/d3Ny4Z8+ePi2hsbGRs2fPplwuZ0JCAg8dOtRj7cOHD/f6/cLCwh7j09PTWVhYKDlBTk5OFEWR4eHhtKTa6yzwVFtbGydOnEiSfOutt3pFWCzRm+UunzBhAltaWjh69GgC4LJly5iXl0dBEOjm5sbg4GA6OTnxgQce4M6dO/u0hGvXrtGCSSQkJPRYWy6XS95id+6OUQJgZmam1Y8jCAJJWhRQpzArYOjJkyehVquhVt9CkH/++ederyeDwQCDwQCVSgVRFFFWVobS0lIkJCRg5cqVWLFiBdasWQMPDw9s2bIFTzzxBGpraxEXF4dly5ZBo9FgxowZkMt74hLe3t5ISUnB888/j23btvXof+ONNzB27Fibz1VWVob333/f6rOGhgZ4enrCzs4Oer0e7u7u0Ov1aGtrA4A6mUKh+K9SqcTTTz+NlpYWNDc3S0L2Ro2NjWhoaEBQUBAAQKfTYdeuXZgyZQoWL14MOzs7HDt2DFqtFjNnzsTcuXMRFhYGPz8/hIaGYvHixdi3b1+v87u5uWHz5s2IjY3t0Tds2DCEhoba5MDAwB7ji4qK4OPjg5CQEMhkMsyZMwcNDQ1obGwEgKtITEzcn5mZyaamJn7zzTfUaDQkyccff7zXk9XR0ZGFhYVctmyZ9NmQIUN45coV3rhxg/v27aNcLmd4eDibm5uZm5vLvXv3sqCggBcvXpSSKdnZ2TavTgvV1tbyySefvKski6urK8vLy1lSUsLdu3ezvr6eqampljk3YsGCBe/n5+fzzTfflCDu0tJSTpw4sc+Jly5dypycHKt9l5SUxNLSUgmalsvlXLp0KY8dO8ZPP/2U0dHRPHPmDAEwODiYPj4+zMjI6PNMuHTpUp8/Rn94+PDh/OKLL1hUVMQ1a9ZIcsrl8iV3FAxZMrm7d+9mWFjYXT2ct7e3lEfo64qMjY0d7HRbq6+v79MDDod7i7/vhv39/Zmdnd2nJVRVVXHy5MmDGg4LghBqExC5H+zv788jR4706Tpfu3aN0dHRg2UJEiAiM9fZ3PdE5vDhw5mbm9unJVRUVDAqKmow1kvpCgNagaL3WwnHjh3r0xKuXLnC3//+93ez/SRQ1AoWDwwM5PDhwwc84eTJkymKIgEwICCAw4YNs+rXaDSSlwiAo0ePpoODQ6/toKAgHjlypE9LKC8vt4LRB8g9YHFoNJqF5eXlpr/+9a8DntCS4rL46l1dz64wu+UX6w/MrdVqefz48T4tobq6mpGRkQO1BKvEiKwLrJ2/bt26ms8++wwAEBAQgGHDhsHNzQ0jRoyAp+etnKK7uztGjBgBb29vm16c0Wjs4UVaYC+S8Pf3BwD4+fkhJCQEo0aNAgD4+/sjJCQEDg4OCAwMRGVlJf785z/jxx9/7NVjDAgIQFpamuQay2Qy+Pj4YOTIkdBqtXB1dZVgOA8PD/j6+kIQhIsACmwmR11cXDZYNPX999+zsrJSMsWysjJOmTKFRUVF0rVkMeuuFvDdd99JlZ8WtmR3ANzWtKdNm0aSfPPNNxkUFMSQkBCeOHHitgfjhAkTqFarmZKSQpPJRKPRyIKCAoaEhDA0NJR5eXkcNWpUn8lRmFPHVZYS1oaGBs6ZM4djx45ldXU1q6urrdobN260qQCj0cj29naJLaixIAi0s7MjSc6cOVNKhHZty2QykuSZM2c4cuRIqlQqjho1isePH+8TaT59+jRHjx5NjUZDZ2dnhoSEMD8/n3v37uVXX33FxMREm+nx7oWSNwC8bb4i1GfPnkVGRgaMRiMqKythb29v1X7wwQdtavHChQv46KOPJKB05MiRWLVqFQBAr9dLf9vb/1fQ2b29bt06lJeXAwCuXr2KxYsXY9u2bXjooYdsIs1hYWHYu3cv5s2bh6qqKjz99NPo6OjA1KlTcfHiRSQkJLSbZbtxu1rhHACFDQ0N0Ov1MBqN0j7u2jYajbC3t7ep###Xr+Pzzz9HamoqUlNTceDAgQHDtR0dHdL/EyZMQGtrK5577jmcPHmy1+9otVps2bIFSqUSxcXF8PLywoULF/D++++jpaWl0CzbbYulGwCsysrKaureMVjwdUdHBwICAnptA8DLL78MX19feHt7IykpCYmJiaioqMCSJUtQWlraa84hIiICr732GnJychAbG4tFixbhm2++uQ5glVm2flWLn9y5c+erhw8f7uwtIdlbctJWX/f2gQMHsGHDBnR0dCA2NrZHGwBu3ryJkpISVFZWIiwsDPv378fIkSNRX1+P5557DiUlJb2uHx0dDZlMhk2bNuHSpUut5iLqkwMulHR0dPy3pWhixowZ/MMf/iCd7NOnT5fa3t7eXLx4cY/Pu0Z8ln4AfOCBBzhv3jwmJCTQz8/Pqu3r6yv5BY8//jgXLlzIRx55hDKZjF5eXlKhxMSJE3nu3Dmbh+LWrVspCAI1Gs1dFUpaSmX/fa/d5O4vTvTG48ePZ3FxsZXwlZWVluvOBODTuymVtZC7ufDYeC8VMG3atH6NHTFiBNPS0piXl8ddu3Z19QwPDGbFuLvZEsR7oQCNRmMT4e2NlUolhwwZYkl2mABsHUzhu26H9ebaW/5GWXe3FeL9qSD/k7n29p4K5+Pjc7uiimvmZ1PjHtAEc+1t271SwKFDh1hQUGAVLU6dOpX+/v7t5me5Jy9NSX4CgLkApgH4ALfe3vpVyWAwQBRFAMCUKVNgNBrxww8/VBsMhrfNHl7DHb0OdBfP1I5b7/juBCAPDg72GjdunLOXl5csMTERoaGhOH36NERRhEajQVxcHOLj4zF58mSIoojLly9Dq9UiOTkZJ06cwMqVK/Hoo4/i7NmzaGtrg4ODA1555RU8++yz0Ov1GDduHI4ePYqSkhJDfX191blz57aaTKYXzNheO+4zyZVK5YMajWahq6vrT4sWLTLl5eUxNTWVLi4uzMnJYW1tLauqqnj58mXW1dVx1qxZUqlsVlYWq6qq2Nrays2bN1OlUnH79u2sqqriwYMH+corr9DLy8uoUCiOC4Lw23p1tpdtFatSqVLHjx9fEhUV9curr77aYYHaVCoVk5KSWFRUxJiYGOr1es6bN49yuZxxcXFcv349PTw8+Mc//rHd09PzuvkXTjFjeIrBflihl/2mkMlkTgAUwi1HXiDZaPbpXVtbW39xcnKiOUByNcf6wi23X2gAgPj4eHl4ePiI4uJij5qaGv+Wlhbftra2EL1erzWZTN4k3a5everk4+ODixcvNpGs02g01wIDA6uvXr16IiIioik/P/9UZ2dnjZ+fX8OVK1dczOuZBEGQ63S6RkdHR5vRGUlPk8nUJJfLOy1tAI2CIOi7j/1/l0eTL0xHMHkAAAAASUVORK5CYII=
  32. // @license MIT
  33. // @compatible chrome
  34. // @compatible firefox
  35. // @compatible edge
  36. // @compatible opera
  37. // @compatible safari
  38. // @connect greasyfork.org
  39. // @connect sleazyfork.org
  40. // @connect github.com
  41. // @connect githubusercontent.com
  42. // @connect openuserjs.org
  43. // @grant GM_addElement
  44. // @grant GM_info
  45. // @grant GM_getValue
  46. // @grant GM_openInTab
  47. // @grant GM_setValue
  48. // @grant GM_registerMenuCommand
  49. // @grant GM_xmlhttpRequest
  50. // @grant GM.addElement
  51. // @grant GM.info
  52. // @grant GM.getValue
  53. // @grant GM.openInTab
  54. // @grant GM.setValue
  55. // @grant GM.registerMenuCommand
  56. // @grant GM.xmlHttpRequest
  57. // @match https://*/*
  58. // @noframes
  59. // @run-at document-start
  60. // ==/UserScript==
  61. (() => {
  62. 'use strict';
  63. /******************************************************************************/
  64. const inIframe = (() => {
  65. try {
  66. return window.self !== window.top;
  67. } catch (e) {
  68. return true;
  69. }
  70. })();
  71. if (inIframe) {
  72. return;
  73. }
  74. let userjs = self.userjs;
  75. /**
  76. * Skip text/plain documents, based on uBlock Origin `vapi.js` file
  77. *
  78. * [source code](https://github.com/gorhill/uBlock/blob/68962453ff6eec7ff109615a738beb8699b9844a/platform/common/vapi.js#L35)
  79. */
  80. if (
  81. (document instanceof Document ||
  82. (document instanceof XMLDocument && document.createElement('div') instanceof HTMLDivElement)) &&
  83. /^text\/html|^application\/(xhtml|xml)/.test(document.contentType || '') === true &&
  84. (self.userjs instanceof Object === false || userjs.UserJS !== true)
  85. ) {
  86. userjs = self.userjs = { UserJS: true };
  87. } else {
  88. console.error('[%cMagic Userscript+%c] %cERROR','color: rgb(29, 155, 240);','','color: rgb(249, 24, 128);', `MIME type is not a document, got "${document.contentType || ''}"`);
  89. }
  90. if (!(typeof userjs === 'object' && userjs.UserJS)) {
  91. return;
  92. }
  93. /** Native implementation exists */
  94. if (window.trustedTypes && window.trustedTypes.createPolicy) window.trustedTypes.createPolicy('default', { createHTML: (string) => string });
  95. /** [i18n directory](https://github.com/magicoflolis/Userscript-Plus/tree/master/src/_locales) */
  96. const translations = {
  97. "ar": {
  98. "createdby": "انشأ من قبل",
  99. "name": "اسم",
  100. "daily_installs": "التثبيت اليومي",
  101. "close": "يغلق",
  102. "filterA": "منقي",
  103. "max": "تحقيق أقصى قدر",
  104. "min": "تصغير",
  105. "search": "يبحث",
  106. "search_placeholder": "بحث في البرامج النصية",
  107. "install": "تثبيت",
  108. "issue": "إصدار جديد",
  109. "version_number": "الإصدار",
  110. "updated": "آخر تحديث",
  111. "total_installs": "إجمالي التثبيت",
  112. "ratings": "التقييمات",
  113. "good": "جيد",
  114. "ok": "جيد",
  115. "bad": "سيء",
  116. "created_date": "تم إنشاؤه",
  117. "redirect": "شوكة دهنية للكبار",
  118. "filter": "تصفية اللغات الأخرى",
  119. "dtime": "عرض المهلة",
  120. "save": "حفظ",
  121. "reset": "إعادة تعيين",
  122. "preview_code": "كود المعاينة",
  123. "saveFile": "احفظ الملف",
  124. "newTab": "علامة تبويب جديدة",
  125. "applies_to": "ينطبق على",
  126. "license": "الترخيص",
  127. "no_license": "لا يوجد",
  128. "antifeatures": "إعلانات",
  129. "userjs_fullscreen": "ملء الشاشة الكاملة التلقائي",
  130. "listing_none": "(لا يوجد)",
  131. "export_config": "تهيئة التصدير",
  132. "export_theme": "تصدير السمة",
  133. "import_config": "استيراد تهيئة الاستيراد",
  134. "import_theme": "استيراد النسق",
  135. "code_size": "حجم الرمز",
  136. "prmpt_css": "التثبيت كأسلوب المستخدم؟",
  137. "userjs_inject": "حقن Userscript+",
  138. "userjs_close": "إغلاق Userscript+",
  139. "userjs_sync": "Sync",
  140. "userjs_autoinject": "Inject on load",
  141. "auto_fetch": "Fetch on load",
  142. "code": "Code",
  143. "metadata": "Metadata",
  144. "preview_metadata": "Preview Metadata",
  145. "recommend_author": "Recommend Author",
  146. "recommend_other": "Recommend Others",
  147. "default_sort": "Default Sort"
  148. },
  149. "de": {
  150. "createdby": "Erstellt von",
  151. "name": "Name",
  152. "daily_installs": "Tägliche Installationen",
  153. "close": "Schließen Sie",
  154. "filterA": "Filter",
  155. "max": "Maximieren Sie",
  156. "min": "minimieren",
  157. "search": "Suche",
  158. "search_placeholder": "Suche nach Userscripts",
  159. "install": "Installieren Sie",
  160. "issue": "Neue Ausgabe",
  161. "version_number": "Version",
  162. "updated": "Zuletzt aktualisiert",
  163. "total_installs": "Installationen insgesamt",
  164. "ratings": "Bewertungen",
  165. "good": "Gut",
  166. "ok": "Okay",
  167. "bad": "Schlecht",
  168. "created_date": "Erstellt",
  169. "redirect": "Greasy Fork für Erwachsene",
  170. "filter": "Andere Sprachen herausfiltern",
  171. "dtime": "Zeitüberschreitung anzeigen",
  172. "save": "Speichern Sie",
  173. "reset": "Zurücksetzen",
  174. "preview_code": "Vorschau Code",
  175. "saveFile": "Datei speichern",
  176. "newTab": "Neue Registerkarte",
  177. "applies_to": "Gilt für",
  178. "license": "Lizenz",
  179. "no_license": "N/A",
  180. "antifeatures": "Antifeatures",
  181. "userjs_fullscreen": "Automatischer Vollbildmodus",
  182. "listing_none": "(Keine)",
  183. "export_config": "Konfig exportieren",
  184. "export_theme": "Thema exportieren",
  185. "import_config": "Konfig importieren",
  186. "import_theme": "Thema importieren",
  187. "code_size": "Code Größe",
  188. "prmpt_css": "Als UserStyle installieren?",
  189. "userjs_inject": "Userscript+ einfügen",
  190. "userjs_close": "Userscript+ schließen",
  191. "userjs_sync": "Sync",
  192. "userjs_autoinject": "Inject on load",
  193. "auto_fetch": "Fetch on load",
  194. "code": "Quelltext",
  195. "metadata": "Metadata",
  196. "preview_metadata": "Preview Metadata",
  197. "recommend_author": "Recommend Author",
  198. "recommend_other": "Recommend Others",
  199. "default_sort": "Default Sort"
  200. },
  201. "en": {
  202. "createdby": "Created by",
  203. "name": "Name",
  204. "daily_installs": "Daily Installs",
  205. "close": "Close",
  206. "filterA": "Filter",
  207. "max": "Maximize",
  208. "min": "Minimize",
  209. "search": "Search",
  210. "search_placeholder": "Search for userscripts",
  211. "install": "Install",
  212. "issue": "New Issue",
  213. "version_number": "Version",
  214. "updated": "Last Updated",
  215. "total_installs": "Total Installs",
  216. "ratings": "Ratings",
  217. "good": "Good",
  218. "ok": "Okay",
  219. "bad": "Bad",
  220. "created_date": "Created",
  221. "redirect": "Greasy Fork for adults",
  222. "filter": "Filter out other languages",
  223. "dtime": "Display Timeout",
  224. "save": "Save",
  225. "reset": "Reset",
  226. "preview_code": "Preview Code",
  227. "saveFile": "Download",
  228. "newTab": "New Tab",
  229. "applies_to": "Applies to",
  230. "license": "License",
  231. "no_license": "N/A",
  232. "antifeatures": "Antifeatures",
  233. "userjs_fullscreen": "Automatic Fullscreen",
  234. "listing_none": "(None)",
  235. "export_config": "Export Config",
  236. "export_theme": "Export Theme",
  237. "import_config": "Import Config",
  238. "import_theme": "Import Theme",
  239. "code_size": "Code Size",
  240. "prmpt_css": "Install as UserStyle?",
  241. "userjs_inject": "Inject Userscript+",
  242. "userjs_close": "Close Userscript+",
  243. "userjs_sync": "Sync",
  244. "userjs_autoinject": "Inject on load",
  245. "auto_fetch": "Fetch on load",
  246. "code": "Code",
  247. "metadata": "Metadata",
  248. "preview_metadata": "Preview Metadata",
  249. "recommend_author": "Recommend Author",
  250. "recommend_other": "Recommend Others",
  251. "default_sort": "Default Sort"
  252. },
  253. "en_GB": {
  254. "createdby": "Created by",
  255. "name": "Name",
  256. "daily_installs": "Daily Installs",
  257. "close": "Close",
  258. "filterA": "Filter",
  259. "max": "Maximize",
  260. "min": "Minimize",
  261. "search": "Search",
  262. "search_placeholder": "Search for userscripts",
  263. "install": "Install",
  264. "issue": "New Issue",
  265. "version_number": "Version",
  266. "updated": "Last Updated",
  267. "total_installs": "Total Installs",
  268. "ratings": "Ratings",
  269. "good": "Good",
  270. "ok": "Okay",
  271. "bad": "Bad",
  272. "created_date": "Created",
  273. "redirect": "Greasy Fork for adults",
  274. "filter": "Filter out other languages",
  275. "dtime": "Display Timeout",
  276. "save": "Save",
  277. "reset": "Reset",
  278. "preview_code": "Preview Code",
  279. "saveFile": "Download",
  280. "newTab": "New Tab",
  281. "applies_to": "Applies to",
  282. "license": "License",
  283. "no_license": "N/A",
  284. "antifeatures": "Antifeatures",
  285. "userjs_fullscreen": "Automatic Fullscreen",
  286. "listing_none": "(None)",
  287. "export_config": "Export Config",
  288. "export_theme": "Export Theme",
  289. "import_config": "Import Config",
  290. "import_theme": "Import Theme",
  291. "code_size": "Code Size",
  292. "prmpt_css": "Install as UserStyle?",
  293. "userjs_inject": "Inject Userscript+",
  294. "userjs_close": "Close Userscript+",
  295. "userjs_sync": "Sync",
  296. "userjs_autoinject": "Inject on load",
  297. "auto_fetch": "Fetch on load",
  298. "code": "Code",
  299. "metadata": "Metadata",
  300. "preview_metadata": "Preview Metadata",
  301. "recommend_author": "Recommend Author",
  302. "recommend_other": "Recommend Others",
  303. "default_sort": "Default Sort"
  304. },
  305. "es": {
  306. "createdby": "Creado por",
  307. "name": "Nombre",
  308. "daily_installs": "Instalaciones diarias",
  309. "close": "Ya no se muestra",
  310. "filterA": "Filtro",
  311. "max": "Maximizar",
  312. "min": "Minimizar",
  313. "search": "Busque en",
  314. "search_placeholder": "Buscar userscripts",
  315. "install": "Instalar",
  316. "issue": "Nueva edición",
  317. "version_number": "Versión",
  318. "updated": "Última actualización",
  319. "total_installs": "Total de instalaciones",
  320. "ratings": "Clasificaciones",
  321. "good": "Bueno",
  322. "ok": "Ok",
  323. "bad": "Malo",
  324. "created_date": "Creado",
  325. "redirect": "Greasy Fork para adultos",
  326. "filter": "Filtrar otros idiomas",
  327. "dtime": "Mostrar el tiempo de espera",
  328. "save": "Guardar",
  329. "reset": "Reiniciar",
  330. "preview_code": "Vista previa del código",
  331. "saveFile": "Guardar archivo",
  332. "newTab": "Guardar archivo",
  333. "applies_to": "Se aplica a",
  334. "license": "Licencia",
  335. "no_license": "Desconocida",
  336. "antifeatures": "Características indeseables",
  337. "userjs_fullscreen": "Pantalla completa automática",
  338. "listing_none": "(Ninguno)",
  339. "export_config": "Exportar configuración",
  340. "export_theme": "Exportar tema",
  341. "import_config": "Importar configuración",
  342. "import_theme": "Importar tema",
  343. "code_size": "Código Tamaño",
  344. "prmpt_css": "¿Instalar como UserStyle?",
  345. "userjs_inject": "Inyectar Userscript+",
  346. "userjs_close": "Cerrar Userscript+",
  347. "userjs_sync": "Sync",
  348. "userjs_autoinject": "Inject on load",
  349. "auto_fetch": "Fetch on load",
  350. "code": "Código",
  351. "metadata": "Metadata",
  352. "preview_metadata": "Preview Metadata",
  353. "recommend_author": "Recommend Author",
  354. "recommend_other": "Recommend Others",
  355. "default_sort": "Default Sort"
  356. },
  357. "fr": {
  358. "createdby": "Créé par",
  359. "name": "Nom",
  360. "daily_installs": "Installations quotidiennes",
  361. "close": "Ne plus montrer",
  362. "filterA": "Filtre",
  363. "max": "Maximiser",
  364. "min": "Minimiser",
  365. "search": "Recherche",
  366. "search_placeholder": "Rechercher des userscripts",
  367. "install": "Installer",
  368. "issue": "Nouveau numéro",
  369. "version_number": "Version",
  370. "updated": "Dernière mise à jour",
  371. "total_installs": "Total des installations",
  372. "ratings": "Notations",
  373. "good": "Bon",
  374. "ok": "Ok",
  375. "bad": "Mauvais",
  376. "created_date": "Créé",
  377. "redirect": "Greasy Fork pour les adultes",
  378. "filter": "Filtrer les autres langues",
  379. "dtime": "Délai d'affichage",
  380. "save": "Sauvez",
  381. "reset": "Réinitialiser",
  382. "preview_code": "Prévisualiser le code",
  383. "saveFile": "Enregistrer le fichier",
  384. "newTab": "Nouvel onglet",
  385. "applies_to": "S'applique à",
  386. "license": "Licence",
  387. "no_license": "N/A",
  388. "antifeatures": "Antifeatures",
  389. "userjs_fullscreen": "Plein écran automatique",
  390. "listing_none": "(Aucun)",
  391. "export_config": "Export Config",
  392. "export_theme": "Exporter le thème",
  393. "import_config": "Importer la configuration",
  394. "import_theme": "Importer le thème",
  395. "code_size": "Code Taille",
  396. "prmpt_css": "Installer comme UserStyle ?",
  397. "userjs_inject": "Injecter Userscript+",
  398. "userjs_close": "Fermer Userscript+",
  399. "userjs_sync": "Sync",
  400. "userjs_autoinject": "Inject on load",
  401. "auto_fetch": "Fetch on load",
  402. "code": "Code",
  403. "metadata": "Metadata",
  404. "preview_metadata": "Preview Metadata",
  405. "recommend_author": "Recommend Author",
  406. "recommend_other": "Recommend Others",
  407. "default_sort": "Default Sort"
  408. },
  409. "ja": {
  410. "createdby": "によって作成された",
  411. "name": "名前",
  412. "daily_installs": "デイリーインストール",
  413. "close": "表示されなくなりました",
  414. "filterA": "フィルター",
  415. "max": "最大化",
  416. "min": "ミニマム",
  417. "search": "検索",
  418. "search_placeholder": "ユーザースクリプトの検索",
  419. "install": "インストール",
  420. "issue": "新刊のご案内",
  421. "version_number": "バージョン",
  422. "updated": "最終更新日",
  423. "total_installs": "総インストール数",
  424. "ratings": "レーティング",
  425. "good": "グッド",
  426. "ok": "良い",
  427. "bad": "悪い",
  428. "created_date": "作成",
  429. "redirect": "大人のGreasyfork",
  430. "filter": "他の言語をフィルタリングする",
  431. "dtime": "表示タイムアウト",
  432. "save": "拯救",
  433. "reset": "リセット",
  434. "preview_code": "コードのプレビュー",
  435. "saveFile": "ファイルを保存",
  436. "newTab": "新しいタブ",
  437. "applies_to": "適用対象",
  438. "license": "ライセンス",
  439. "no_license": "不明",
  440. "antifeatures": "アンチ機能",
  441. "userjs_fullscreen": "自動フルスクリーン",
  442. "listing_none": "(なし)",
  443. "export_config": "エクスポート設定",
  444. "export_theme": "テーマのエクスポート",
  445. "import_config": "設定のインポート",
  446. "import_theme": "テーマのインポート",
  447. "code_size": "コード・サイズ",
  448. "prmpt_css": "UserStyleとしてインストールしますか?",
  449. "userjs_inject": "Userscript+ を挿入",
  450. "userjs_close": "Userscript+ を閉じる",
  451. "userjs_sync": "Sync",
  452. "userjs_autoinject": "Inject on load",
  453. "auto_fetch": "Fetch on load",
  454. "code": "コード",
  455. "metadata": "Metadata",
  456. "preview_metadata": "Preview Metadata",
  457. "recommend_author": "Recommend Author",
  458. "recommend_other": "Recommend Others",
  459. "default_sort": "Default Sort"
  460. },
  461. "nl": {
  462. "createdby": "Gemaakt door",
  463. "name": "Naam",
  464. "daily_installs": "Dagelijkse Installaties",
  465. "close": "Sluit",
  466. "filterA": "Filter",
  467. "max": "Maximaliseer",
  468. "min": "Minimaliseer",
  469. "search": "Zoek",
  470. "search_placeholder": "Zoeken naar gebruikersscripts",
  471. "install": "Installeer",
  472. "issue": "Nieuw Issue",
  473. "version_number": "Versie",
  474. "updated": "Laatste Update",
  475. "total_installs": "Totale Installaties",
  476. "ratings": "Beoordeling",
  477. "good": "Goed",
  478. "ok": "Ok",
  479. "bad": "Slecht",
  480. "created_date": "Aangemaakt",
  481. "redirect": "Greasy Fork voor volwassenen",
  482. "filter": "Filter andere talen",
  483. "dtime": "Weergave timeout",
  484. "save": "Opslaan",
  485. "reset": "Opnieuw instellen",
  486. "preview_code": "Voorbeeldcode",
  487. "saveFile": "Bestand opslaan",
  488. "newTab": "Nieuw tabblad",
  489. "applies_to": "Geldt voor",
  490. "license": "Licentie",
  491. "no_license": "N.v.t.",
  492. "antifeatures": "Functies voor eigen gewin",
  493. "userjs_fullscreen": "Automatisch volledig scherm",
  494. "listing_none": "(Geen)",
  495. "export_config": "Configuratie exporteren",
  496. "export_theme": "Thema exporteren",
  497. "import_config": "Configuratie importeren",
  498. "import_theme": "Thema importeren",
  499. "code_size": "Code Grootte",
  500. "prmpt_css": "Installeren als UserStyle?",
  501. "userjs_inject": "Injecteer Userscript+",
  502. "userjs_close": "Sluit Userscript+",
  503. "userjs_sync": "Sync",
  504. "userjs_autoinject": "Inject on load",
  505. "auto_fetch": "Fetch on load",
  506. "code": "Code",
  507. "metadata": "Metadata",
  508. "preview_metadata": "Preview Metadata",
  509. "recommend_author": "Recommend Author",
  510. "recommend_other": "Recommend Others",
  511. "default_sort": "Default Sort"
  512. },
  513. "pl": {
  514. "createdby": "Stworzony przez",
  515. "name": "Nazwa",
  516. "daily_installs": "Codzienne instalacje",
  517. "close": "Zamknij",
  518. "filterA": "Filtr",
  519. "max": "Maksymalizuj",
  520. "min": "Minimalizuj",
  521. "search": "Wyszukiwanie",
  522. "search_placeholder": "Wyszukiwanie skryptów użytkownika",
  523. "install": "Instalacja",
  524. "issue": "Nowy numer",
  525. "version_number": "Wersja",
  526. "updated": "Ostatnia aktualizacja",
  527. "total_installs": "Łączna liczba instalacji",
  528. "ratings": "Oceny",
  529. "good": "Dobry",
  530. "ok": "Ok",
  531. "bad": "Zły",
  532. "created_date": "Utworzony",
  533. "redirect": "Greasy Fork dla dorosłych",
  534. "filter": "Odfiltruj inne języki",
  535. "dtime": "Limit czasu wyświetlania",
  536. "save": "Zapisz",
  537. "reset": "Reset",
  538. "preview_code": "Kod podglądu",
  539. "saveFile": "Zapisz plik",
  540. "newTab": "Nowa karta",
  541. "applies_to": "Dotyczy",
  542. "license": "Licencja",
  543. "no_license": "N/A",
  544. "antifeatures": "Antywzorce",
  545. "userjs_fullscreen": "Automatyczny pełny ekran",
  546. "listing_none": "(Brak)",
  547. "export_config": "Konfiguracja eksportu",
  548. "export_theme": "Motyw eksportu",
  549. "import_config": "Importuj konfigurację",
  550. "import_theme": "Importuj motyw",
  551. "code_size": "Kod Rozmiar",
  552. "prmpt_css": "Zainstalować jako UserStyle?",
  553. "userjs_inject": "Wstrzyknij Userscript+",
  554. "userjs_close": "Zamknij Userscript+",
  555. "userjs_sync": "Sync",
  556. "userjs_autoinject": "Inject on load",
  557. "auto_fetch": "Fetch on load",
  558. "code": "Kod",
  559. "metadata": "Metadata",
  560. "preview_metadata": "Preview Metadata",
  561. "recommend_author": "Recommend Author",
  562. "recommend_other": "Recommend Others",
  563. "default_sort": "Default Sort"
  564. },
  565. "ru": {
  566. "createdby": "Сделано",
  567. "name": "Имя",
  568. "daily_installs": "Ежедневные установки",
  569. "close": "Больше не показывать",
  570. "filterA": "Фильтр",
  571. "max": "Максимизировать",
  572. "min": "Минимизировать",
  573. "search": "Поиск",
  574. "search_placeholder": "Поиск юзерскриптов",
  575. "install": "Установите",
  576. "issue": "Новый выпуск",
  577. "version_number": "Версия",
  578. "updated": "Последнее обновление",
  579. "total_installs": "Всего установок",
  580. "ratings": "Рейтинги",
  581. "good": "Хорошо",
  582. "ok": "Хорошо",
  583. "bad": "Плохо",
  584. "created_date": "Создано",
  585. "redirect": "Greasy Fork для взрослых",
  586. "filter": "Отфильтровать другие языки",
  587. "dtime": "Тайм-аут отображения",
  588. "save": "Сохранить",
  589. "reset": "Перезагрузить",
  590. "preview_code": "Предварительный просмотр кода",
  591. "saveFile": "Сохранить файл",
  592. "newTab": "Новая вкладка",
  593. "applies_to": "Применяется к",
  594. "license": "Лицензия",
  595. "no_license": "Недоступно",
  596. "antifeatures": "Нежелательная функциональность",
  597. "userjs_fullscreen": "Автоматический полноэкранный режим",
  598. "listing_none": "(нет)",
  599. "export_config": "Экспорт конфигурации",
  600. "export_theme": "Экспорт темы",
  601. "import_config": "Импорт конфигурации",
  602. "import_theme": "Импортировать тему",
  603. "code_size": "Код Размер",
  604. "prmpt_css": "Установить как UserStyle?",
  605. "userjs_inject": "Вставить Userscript+",
  606. "userjs_close": "Закрыть Userscript+",
  607. "userjs_sync": "Sync",
  608. "userjs_autoinject": "Inject on load",
  609. "auto_fetch": "Fetch on load",
  610. "code": "Исходный код",
  611. "metadata": "Metadata",
  612. "preview_metadata": "Preview Metadata",
  613. "recommend_author": "Recommend Author",
  614. "recommend_other": "Recommend Others",
  615. "default_sort": "Default Sort"
  616. },
  617. "zh": {
  618. "createdby": "由...制作",
  619. "name": "姓名",
  620. "daily_installs": "日常安装",
  621. "close": "不再显示",
  622. "filterA": "过滤器",
  623. "max": "最大化",
  624. "min": "最小化",
  625. "search": "搜索",
  626. "search_placeholder": "搜索用户脚本",
  627. "install": "安装",
  628. "issue": "新问题",
  629. "version_number": "版本",
  630. "updated": "最后更新",
  631. "total_installs": "总安装量",
  632. "ratings": "评级",
  633. "good": "好的",
  634. "ok": "好的",
  635. "bad": "不好",
  636. "created_date": "创建",
  637. "redirect": "大人的Greasyfork",
  638. "filter": "过滤掉其他语言",
  639. "dtime": "显示超时",
  640. "save": "拯救",
  641. "reset": "重置",
  642. "preview_code": "预览代码",
  643. "saveFile": "保存存档",
  644. "newTab": "新标签",
  645. "applies_to": "适用于",
  646. "license": "许可证",
  647. "no_license": "暂无",
  648. "antifeatures": "可能不受欢迎的功能",
  649. "userjs_fullscreen": "自动全屏",
  650. "listing_none": "(无)",
  651. "export_config": "导出配置",
  652. "export_theme": "导出主题",
  653. "import_config": "导入配置",
  654. "import_theme": "导入主题",
  655. "code_size": "代码 尺寸",
  656. "prmpt_css": "安装为用户风格?",
  657. "userjs_inject": "注入 Userscript+",
  658. "userjs_close": "关闭 Userscript+",
  659. "userjs_sync": "Sync",
  660. "userjs_autoinject": "Inject on load",
  661. "auto_fetch": "Fetch on load",
  662. "code": "代码",
  663. "metadata": "Metadata",
  664. "preview_metadata": "Preview Metadata",
  665. "recommend_author": "Recommend Author",
  666. "recommend_other": "Recommend Others",
  667. "default_sort": "Default Sort"
  668. },
  669. "zh_CN": {
  670. "createdby": "由...制作",
  671. "name": "姓名",
  672. "daily_installs": "日常安装",
  673. "close": "不再显示",
  674. "filterA": "过滤器",
  675. "max": "最大化",
  676. "min": "最小化",
  677. "search": "搜索",
  678. "search_placeholder": "搜索用户脚本",
  679. "install": "安装",
  680. "issue": "新问题",
  681. "version_number": "版本",
  682. "updated": "最后更新",
  683. "total_installs": "总安装量",
  684. "ratings": "评级",
  685. "good": "好的",
  686. "ok": "好的",
  687. "bad": "不好",
  688. "created_date": "创建",
  689. "redirect": "大人的Greasyfork",
  690. "filter": "过滤掉其他语言",
  691. "dtime": "显示超时",
  692. "save": "拯救",
  693. "reset": "重置",
  694. "preview_code": "预览代码",
  695. "saveFile": "保存存档",
  696. "newTab": "新标签",
  697. "applies_to": "适用于",
  698. "license": "许可证",
  699. "no_license": "暂无",
  700. "antifeatures": "可能不受欢迎的功能",
  701. "userjs_fullscreen": "自动全屏",
  702. "listing_none": "(无)",
  703. "export_config": "导出配置",
  704. "export_theme": "导出主题",
  705. "import_config": "导入配置",
  706. "import_theme": "导入主题",
  707. "code_size": "代码 尺寸",
  708. "prmpt_css": "安装为用户风格?",
  709. "userjs_inject": "注入 Userscript+",
  710. "userjs_close": "关闭 Userscript+",
  711. "userjs_sync": "Sync",
  712. "userjs_autoinject": "Inject on load",
  713. "auto_fetch": "Fetch on load",
  714. "code": "代码",
  715. "metadata": "Metadata",
  716. "preview_metadata": "Preview Metadata",
  717. "recommend_author": "Recommend Author",
  718. "recommend_other": "Recommend Others",
  719. "default_sort": "Default Sort"
  720. },
  721. "zh_TW": {
  722. "createdby": "由...制作",
  723. "name": "姓名",
  724. "daily_installs": "日常安装",
  725. "close": "不再显示",
  726. "filterA": "过滤器",
  727. "max": "最大化",
  728. "min": "最小化",
  729. "search": "搜索",
  730. "search_placeholder": "搜索用户脚本",
  731. "install": "安装",
  732. "issue": "新问题",
  733. "version_number": "版本",
  734. "updated": "最后更新",
  735. "total_installs": "总安装量",
  736. "ratings": "评级",
  737. "good": "好的",
  738. "ok": "好的",
  739. "bad": "不好",
  740. "created_date": "创建",
  741. "redirect": "大人的Greasyfork",
  742. "filter": "过滤掉其他语言",
  743. "dtime": "显示超时",
  744. "save": "拯救",
  745. "reset": "重置",
  746. "preview_code": "预览代码",
  747. "saveFile": "保存存档",
  748. "newTab": "新标签",
  749. "applies_to": "适用于",
  750. "license": "许可证",
  751. "no_license": "暂无",
  752. "antifeatures": "可能不受欢迎的功能",
  753. "userjs_fullscreen": "自动全屏",
  754. "listing_none": "(无)",
  755. "export_config": "导出配置",
  756. "export_theme": "导出主题",
  757. "import_config": "导入配置",
  758. "import_theme": "导入主题",
  759. "code_size": "代码 尺寸",
  760. "prmpt_css": "作為使用者樣式安裝?",
  761. "userjs_inject": "注入用戶腳本+",
  762. "userjs_close": "關閉用戶腳本+",
  763. "userjs_sync": "Sync",
  764. "userjs_autoinject": "Inject on load",
  765. "auto_fetch": "Fetch on load",
  766. "code": "代碼",
  767. "metadata": "Metadata",
  768. "preview_metadata": "Preview Metadata",
  769. "recommend_author": "Recommend Author",
  770. "recommend_other": "Recommend Others",
  771. "default_sort": "Default Sort"
  772. }
  773. };
  774. /** [source code](https://github.com/magicoflolis/Userscript-Plus/blob/master/src/sass/_main.scss) */
  775. const main_css = `mujs-root {
  776. --mujs-even-row: hsl(222, 14%, 22%);
  777. --mujs-odd-row: hsl(222, 14%, 11%);
  778. --mujs-even-err: hsl(0, 100%, 22%);
  779. --mujs-odd-err: hsl(0, 100%, 11%);
  780. --mujs-background-color: hsl(222, 14%, 33%);
  781. --mujs-gf-color: hsl(204, 100%, 40%);
  782. --mujs-sf-color: hsl(12, 86%, 50%);
  783. --mujs-border-b-color: hsla(0, 0%, 0%, 0);
  784. --mujs-gf-btn-color: hsl(211, 87%, 56%);
  785. --mujs-sf-btn-color: hsl(12, 86%, 50%);
  786. --mujs-sf-txt-color: hsl(12, 79%, 55%);
  787. --mujs-txt-color: hsl(0, 0%, 100%);
  788. --mujs-chck-color: hsla(0, 0%, 100%, 0.568);
  789. --mujs-chck-gf: hsla(197, 100%, 50%, 0.568);
  790. --mujs-chck-git: hsla(213, 13%, 16%, 0.568);
  791. --mujs-chck-open: hsla(12, 86%, 50%, 0.568);
  792. --mujs-placeholder: hsl(81, 56%, 54%);
  793. --mujs-position-top: unset;
  794. --mujs-position-bottom: 1em;
  795. --mujs-position-left: unset;
  796. --mujs-position-right: 1em;
  797. --mujs-font-family: Arial, Helvetica, sans-serif;
  798. font-family: var(--mujs-font-family, Arial, Helvetica, sans-serif);
  799. text-rendering: optimizeLegibility;
  800. word-break: normal;
  801. font-size: 14px;
  802. color: var(--mujs-txt-color, hsl(0, 0%, 100%));
  803. }
  804. mujs-root * {
  805. -webkit-appearance: none;
  806. -moz-appearance: none;
  807. appearance: none;
  808. scrollbar-color: var(--mujs-txt-color, hsl(0, 0%, 100%)) hsl(224, 14%, 21%);
  809. scrollbar-width: thin;
  810. }
  811. @supports not (scrollbar-width: thin) {
  812. mujs-root * ::-webkit-scrollbar {
  813. width: 1.4vw;
  814. height: 3.3vh;
  815. }
  816. mujs-root * ::-webkit-scrollbar-track {
  817. background-color: hsl(224, 14%, 21%);
  818. border-radius: 16px;
  819. margin-top: 3px;
  820. margin-bottom: 3px;
  821. box-shadow: inset 0 0 6px hsla(0, 0%, 0%, 0.3);
  822. }
  823. mujs-root * ::-webkit-scrollbar-thumb {
  824. border-radius: 16px;
  825. background-color: var(--mujs-txt-color, hsl(0, 0%, 100%));
  826. 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);
  827. }
  828. mujs-root * ::-webkit-scrollbar-thumb:hover {
  829. background: var(--mujs-txt-color, hsl(0, 0%, 100%));
  830. }
  831. }
  832. mu-js {
  833. line-height: normal;
  834. }
  835. mujs-section > label,
  836. .mujs-homepag e,
  837. td.mujs-list,
  838. .install {
  839. font-size: 16px;
  840. }
  841. .install,
  842. .mujs-homepage {
  843. font-weight: 700;
  844. }
  845. mujs-section > label,
  846. td.mujs-list {
  847. font-weight: 500;
  848. }
  849. .mujs-invalid {
  850. border-radius: 8px !important;
  851. border-width: 2px !important;
  852. border-style: solid !important;
  853. border-color: hsl(0, 100%, 50%) !important;
  854. }
  855. mujs-tabs,
  856. mujs-column,
  857. mujs-row,
  858. .mujs-sty-flex {
  859. display: flex;
  860. }
  861. mujs-column,
  862. mujs-row {
  863. gap: 0.5em;
  864. }
  865. mujs-column count-frame[data-counter=greasyfork] {
  866. background: var(--mujs-gf-color, hsl(204, 100%, 40%));
  867. }
  868. mujs-column count-frame[data-counter=sleazyfork] {
  869. background: var(--mujs-sf-color, hsl(12, 86%, 50%));
  870. }
  871. mujs-column count-frame[data-counter=github] {
  872. background: hsl(213, 13%, 16%);
  873. }
  874. mujs-column count-frame[data-counter=openuserjs] {
  875. background: hsla(12, 86%, 50%, 0.568);
  876. }
  877. @media screen and (max-width: 800px) {
  878. mujs-column {
  879. flex-flow: row wrap;
  880. }
  881. }
  882. mujs-row {
  883. flex-flow: column wrap;
  884. }
  885. mu-js {
  886. cursor: default;
  887. }
  888. .hidden {
  889. display: none !important;
  890. z-index: -1 !important;
  891. }
  892. mujs-main {
  893. width: 100%;
  894. width: -moz-available;
  895. width: -webkit-fill-available;
  896. background: var(--mujs-background-color, hsl(222, 14%, 33%)) !important;
  897. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  898. border-radius: 16px;
  899. }
  900. @media screen and (max-height: 720px) {
  901. mujs-main:not(.webext-page) {
  902. height: 100% !important;
  903. bottom: 0rem !important;
  904. right: 0rem !important;
  905. margin: 0rem !important;
  906. }
  907. }
  908. mujs-main.expanded {
  909. height: 100% !important;
  910. bottom: 0rem !important;
  911. }
  912. mujs-main:not(.webext-page) {
  913. position: fixed;
  914. height: 492px;
  915. }
  916. mujs-main:not(.webext-page):not(.expanded) {
  917. margin-left: 1rem;
  918. margin-right: 1rem;
  919. right: 1rem;
  920. bottom: 1rem;
  921. }
  922. mujs-main:not(.hidden) {
  923. z-index: 100000000000000000 !important;
  924. display: flex !important;
  925. flex-direction: column !important;
  926. }
  927. mujs-main > * {
  928. width: 100%;
  929. width: -moz-available;
  930. width: -webkit-fill-available;
  931. }
  932. mujs-main mujs-toolbar {
  933. order: 0;
  934. padding: 0.5em;
  935. display: flex;
  936. place-content: space-between;
  937. }
  938. mujs-main mujs-toolbar mujs-tabs {
  939. overflow: hidden;
  940. order: 0;
  941. }
  942. mujs-main mujs-toolbar mujs-column {
  943. flex-flow: row nowrap;
  944. order: 999999999999;
  945. }
  946. mujs-main mujs-toolbar > * {
  947. width: -webkit-fit-content;
  948. width: -moz-fit-content;
  949. width: fit-content;
  950. }
  951. mujs-main mujs-tabs {
  952. gap: 0.5em;
  953. text-align: center;
  954. -webkit-user-select: none;
  955. -moz-user-select: none;
  956. -ms-user-select: none;
  957. user-select: none;
  958. flex-flow: row wrap;
  959. }
  960. mujs-main mujs-tabs mujs-tab {
  961. padding: 0.25em;
  962. min-width: 150px;
  963. width: -webkit-fit-content;
  964. width: -moz-fit-content;
  965. width: fit-content;
  966. height: -webkit-fit-content;
  967. height: -moz-fit-content;
  968. height: fit-content;
  969. display: flex;
  970. place-content: space-between;
  971. border: 1px solid transparent;
  972. border-radius: 4px;
  973. background: transparent;
  974. }
  975. @media screen and (max-width: 800px) {
  976. mujs-main mujs-tabs mujs-tab {
  977. min-width: 6em !important;
  978. }
  979. }
  980. mujs-main mujs-tabs mujs-tab.active {
  981. background: var(--mujs-even-row, hsl(222, 14%, 18%));
  982. }
  983. mujs-main mujs-tabs mujs-tab:not(.active):hover {
  984. background: var(--mujs-even-row, hsl(222, 14%, 18%));
  985. }
  986. mujs-main mujs-tabs mujs-tab mujs-host {
  987. float: left;
  988. overflow: auto;
  989. overflow-wrap: break-word;
  990. text-overflow: ellipsis;
  991. white-space: nowrap;
  992. }
  993. mujs-main mujs-tabs mujs-tab mu-js {
  994. float: right;
  995. }
  996. mujs-main mujs-tabs mujs-addtab {
  997. order: 999999999999;
  998. font-size: 20px;
  999. padding: 0px 0.25em;
  1000. }
  1001. mujs-main mujs-tabs mujs-addtab:hover {
  1002. background: var(--mujs-even-row, hsl(222, 14%, 18%));
  1003. }
  1004. mujs-main mujs-tab,
  1005. mujs-main mujs-btn,
  1006. mujs-main input {
  1007. width: -webkit-fit-content;
  1008. width: -moz-fit-content;
  1009. width: fit-content;
  1010. height: -webkit-fit-content;
  1011. height: -moz-fit-content;
  1012. height: fit-content;
  1013. }
  1014. mujs-main input {
  1015. background: hsla(0, 0%, 0%, 0);
  1016. color: var(--mujs-txt-color, hsl(0, 0%, 100%));
  1017. }
  1018. mujs-main input:not([type=checkbox]) {
  1019. border: transparent;
  1020. outline: none !important;
  1021. }
  1022. mujs-main mujs-page,
  1023. mujs-main textarea {
  1024. background: inherit;
  1025. overflow-y: auto;
  1026. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1027. border-radius: 5px;
  1028. outline: none;
  1029. font-family: monospace;
  1030. font-size: 14px;
  1031. }
  1032. mujs-main mujs-page {
  1033. padding: 0.5em;
  1034. margin: 0.5em;
  1035. }
  1036. mujs-main textarea {
  1037. overflow-y: auto;
  1038. color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1039. resize: vertical;
  1040. }
  1041. mujs-main textarea:focus {
  1042. outline: none;
  1043. }
  1044. mujs-main th,
  1045. mujs-main .mujs-cfg *:not(input[type=password], input[type=text], input[type=number]) {
  1046. -webkit-user-select: none !important;
  1047. -moz-user-select: none !important;
  1048. -ms-user-select: none !important;
  1049. user-select: none !important;
  1050. }
  1051. mujs-main .mujs-footer {
  1052. order: 3;
  1053. overflow-x: hidden;
  1054. text-align: center;
  1055. border-radius: 16px;
  1056. }
  1057. mujs-main .mujs-footer > * {
  1058. min-height: 50px;
  1059. }
  1060. mujs-main .mujs-footer .error:nth-child(even) {
  1061. background: var(--mujs-even-err, hsl(0, 100%, 22%)) !important;
  1062. }
  1063. mujs-main .mujs-footer .error:nth-child(odd) {
  1064. background: var(--mujs-odd-err, hsl(0, 100%, 11%)) !important;
  1065. }
  1066. mujs-main .mujs-prompt {
  1067. align-items: center;
  1068. justify-content: center;
  1069. }
  1070. mujs-main .mujs-prompt svg {
  1071. width: 14px;
  1072. height: 14px;
  1073. background: transparent;
  1074. }
  1075. mujs-main .mujs-prompt > .prompt {
  1076. position: absolute;
  1077. background: var(--mujs-background-color, hsl(222, 14%, 33%)) !important;
  1078. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1079. border-radius: 16px;
  1080. text-align: center;
  1081. padding: 0.5em;
  1082. z-index: 1;
  1083. }
  1084. mujs-main .mujs-prompt > .prompt .prompt-head {
  1085. font-size: 18px;
  1086. }
  1087. mujs-main .mujs-prompt > .prompt .prompt-body {
  1088. display: grid;
  1089. grid-auto-flow: column;
  1090. grid-gap: 0.5em;
  1091. padding-top: 0.5em;
  1092. }
  1093. mujs-main .mujs-prompt > .prompt mujs-btn.prompt-deny {
  1094. background: var(--mujs-sf-btn-color, hsl(12, 86%, 50%));
  1095. border-color: var(--mujs-sf-btn-color, hsl(12, 86%, 50%));
  1096. }
  1097. mujs-main .mujs-prompt > .prompt mujs-btn.prompt-deny:hover {
  1098. background: var(--mujs-sf-txt-color, hsl(12, 79%, 55%));
  1099. border-color: var(--mujs-sf-txt-color, hsl(12, 79%, 55%));
  1100. }
  1101. mujs-main .mujs-prompt > .prompt mujs-btn.prompt-confirm {
  1102. background: var(--mujs-gf-color, hsl(204, 100%, 40%));
  1103. border-color: var(--mujs-gf-color, hsl(204, 100%, 40%));
  1104. }
  1105. mujs-main .mujs-prompt > .prompt mujs-btn.prompt-confirm:hover {
  1106. background: var(--mujs-gf-btn-color, hsl(211, 87%, 56%));
  1107. border-color: var(--mujs-gf-btn-color, hsl(211, 87%, 56%));
  1108. }
  1109. .mainframe {
  1110. background: transparent;
  1111. position: fixed;
  1112. bottom: var(--mujs-position-bottom, 1rem);
  1113. right: var(--mujs-position-right, 1rem);
  1114. top: var(--mujs-position-top, unset);
  1115. left: var(--mujs-position-left, unset);
  1116. }
  1117. .mainframe count-frame {
  1118. width: fit-content;
  1119. width: -moz-fit-content;
  1120. width: -webkit-fit-content;
  1121. height: auto;
  1122. padding: 14px 16px;
  1123. }
  1124. .mainframe.error {
  1125. opacity: 1 !important;
  1126. }
  1127. .mainframe.error count-frame {
  1128. background: var(--mujs-even-err, hsl(0, 100%, 22%)) !important;
  1129. }
  1130. .mainframe:not(.hidden) {
  1131. z-index: 100000000000000000 !important;
  1132. display: block;
  1133. }
  1134. count-frame {
  1135. border-radius: 1000px;
  1136. margin: 0px 3px;
  1137. padding: 4px 6px;
  1138. border: 2px solid var(--mujs-border-b-color, hsla(0, 0%, 0%, 0));
  1139. font-size: 16px;
  1140. font-weight: 400;
  1141. display: inline-block;
  1142. text-align: center;
  1143. min-width: 1em;
  1144. background: var(--mujs-background-color, hsl(222, 14%, 33%));
  1145. -webkit-user-select: none;
  1146. -moz-user-select: none;
  1147. -ms-user-select: none;
  1148. user-select: none;
  1149. }
  1150. mujs-header {
  1151. order: 1;
  1152. display: flex;
  1153. border-bottom: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1154. padding-left: 0.5em;
  1155. padding-right: 0.5em;
  1156. padding-bottom: 0.5em;
  1157. font-size: 1em;
  1158. place-content: space-between;
  1159. height: fit-content;
  1160. height: -moz-fit-content;
  1161. height: -webkit-fit-content;
  1162. gap: 1em;
  1163. }
  1164. mujs-header > *:not(mujs-url) {
  1165. height: fit-content;
  1166. height: -moz-fit-content;
  1167. height: -webkit-fit-content;
  1168. }
  1169. mujs-header mujs-url {
  1170. order: 0;
  1171. flex-grow: 1;
  1172. }
  1173. mujs-header mujs-url > input {
  1174. width: 100%;
  1175. height: 100%;
  1176. background: var(--mujs-even-row, hsl(222, 14%, 18%));
  1177. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1178. border-radius: 4px;
  1179. }
  1180. mujs-header .rate-container {
  1181. order: 1;
  1182. }
  1183. mujs-header .btn-frame {
  1184. order: 999999999999;
  1185. }
  1186. mujs-body {
  1187. order: 2;
  1188. overflow-x: hidden;
  1189. padding: 0px;
  1190. height: 100%;
  1191. border: 1px solid var(--mujs-border-b-color, hsla(0, 0%, 0%, 0));
  1192. border-bottom-left-radius: 16px;
  1193. border-bottom-right-radius: 16px;
  1194. }
  1195. mujs-body .mujs-ratings {
  1196. padding: 0 0.25em;
  1197. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1198. border-radius: 1000px;
  1199. width: -webkit-fit-content;
  1200. width: -moz-fit-content;
  1201. width: fit-content;
  1202. }
  1203. mujs-body mu-jsbtn {
  1204. -webkit-user-select: none;
  1205. -moz-user-select: none;
  1206. -ms-user-select: none;
  1207. user-select: none;
  1208. }
  1209. mujs-body table,
  1210. mujs-body th,
  1211. mujs-body td {
  1212. border-collapse: collapse;
  1213. }
  1214. mujs-body table {
  1215. width: 100%;
  1216. width: -moz-available;
  1217. width: -webkit-fill-available;
  1218. }
  1219. @media screen and (max-width: 1180px) {
  1220. mujs-body table thead > tr {
  1221. display: table-column;
  1222. }
  1223. mujs-body table .frame:not(.webext-page) {
  1224. width: 100%;
  1225. display: flex;
  1226. flex-flow: row wrap;
  1227. align-items: center;
  1228. padding-top: 0.5em;
  1229. padding-bottom: 0.5em;
  1230. }
  1231. mujs-body table .frame:not(.webext-page) td {
  1232. margin: auto;
  1233. }
  1234. mujs-body table .frame:not(.webext-page) td > mujs-a,
  1235. mujs-body table .frame:not(.webext-page) td > mu-js,
  1236. mujs-body table .frame:not(.webext-page) td > mujs-column {
  1237. text-align: center;
  1238. justify-content: center;
  1239. }
  1240. mujs-body table .frame:not(.webext-page) td > mujs-a {
  1241. width: 100%;
  1242. }
  1243. }
  1244. @media screen and (max-width: 1180px) and (max-width: 800px) {
  1245. mujs-body table .frame:not(.webext-page) td > mujs-column {
  1246. flex-flow: column wrap;
  1247. }
  1248. mujs-body table .frame:not(.webext-page) td > mujs-column > mujs-row {
  1249. align-content: center;
  1250. }
  1251. mujs-body table .frame:not(.webext-page) td > mujs-column mujs-column {
  1252. justify-content: center;
  1253. }
  1254. }
  1255. @media screen and (max-width: 1180px) {
  1256. mujs-body table .frame:not(.webext-page) td:not(.mujs-name, .install-btn) {
  1257. width: 25%;
  1258. }
  1259. }
  1260. @media screen and (max-width: 1180px) and (max-width: 800px) {
  1261. mujs-body table .frame:not(.webext-page) td.install-btn {
  1262. width: 100%;
  1263. }
  1264. }
  1265. @media screen and (max-width: 1180px) {
  1266. mujs-body table .frame:not(.webext-page) .mujs-name {
  1267. width: 100%;
  1268. }
  1269. }
  1270. @media screen and (max-width: 550px) {
  1271. mujs-body table .frame:not(.webext-page) td {
  1272. margin: 1rem !important;
  1273. }
  1274. mujs-body table .frame:not(.webext-page) td:not(.mujs-name, .install-btn) {
  1275. width: auto !important;
  1276. }
  1277. }
  1278. mujs-body table th {
  1279. position: -webkit-sticky;
  1280. position: sticky;
  1281. top: 0;
  1282. background: hsla(222, 14%, 33%, 0.75);
  1283. border-bottom: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1284. }
  1285. mujs-body table th.mujs-header-name {
  1286. width: 50%;
  1287. }
  1288. @media screen and (max-width: 800px) {
  1289. mujs-body table th.mujs-header-name {
  1290. width: auto !important;
  1291. }
  1292. }
  1293. mujs-body table .frame:nth-child(even) {
  1294. background: var(--mujs-even-row, hsl(222, 14%, 18%)) !important;
  1295. }
  1296. mujs-body table .frame:nth-child(even) textarea {
  1297. background: var(--mujs-odd-row, hsl(222, 14%, 33%)) !important;
  1298. }
  1299. mujs-body table .frame:nth-child(odd) {
  1300. background: var(--mujs-odd-row, hsl(222, 14%, 33%)) !important;
  1301. }
  1302. mujs-body table .frame:nth-child(odd) textarea {
  1303. background: var(--mujs-even-row, hsl(222, 14%, 18%)) !important;
  1304. }
  1305. mujs-body table .frame:not([data-engine=sleazyfork], [data-engine=greasyfork]) mujs-a {
  1306. color: var(--mujs-sf-txt-color, hsl(12, 79%, 55%));
  1307. }
  1308. mujs-body table .frame:not([data-engine=sleazyfork], [data-engine=greasyfork]) mu-jsbtn {
  1309. background: var(--mujs-sf-btn-color, hsl(12, 86%, 50%));
  1310. border-color: var(--mujs-sf-btn-color, hsl(12, 86%, 50%));
  1311. }
  1312. mujs-body table .frame:not([data-engine=sleazyfork], [data-engine=greasyfork]) mu-jsbtn:hover {
  1313. background: var(--mujs-sf-txt-color, hsl(12, 79%, 55%));
  1314. border-color: var(--mujs-sf-txt-color, hsl(12, 79%, 55%));
  1315. }
  1316. mujs-body table .frame[data-engine=sleazyfork] mujs-a, mujs-body table .frame[data-engine=greasyfork] mujs-a {
  1317. color: var(--mujs-gf-color, hsl(197, 100%, 50%));
  1318. }
  1319. mujs-body table .frame[data-engine=sleazyfork] mujs-a:hover, mujs-body table .frame[data-engine=greasyfork] mujs-a:hover {
  1320. color: var(--mujs-gf-btn-color, hsl(211, 87%, 56%));
  1321. }
  1322. mujs-body table .frame[data-engine=sleazyfork] mu-jsbtn, mujs-body table .frame[data-engine=greasyfork] mu-jsbtn {
  1323. background: var(--mujs-gf-color, hsl(204, 100%, 40%));
  1324. border-color: var(--mujs-gf-color, hsl(204, 100%, 40%));
  1325. }
  1326. mujs-body table .frame[data-engine=sleazyfork] mu-jsbtn:hover, mujs-body table .frame[data-engine=greasyfork] mu-jsbtn:hover {
  1327. background: var(--mujs-gf-btn-color, hsl(211, 87%, 56%));
  1328. border-color: var(--mujs-gf-btn-color, hsl(211, 87%, 56%));
  1329. }
  1330. mujs-body table .frame[data-good] mujs-a, mujs-body table .frame[data-author] mujs-a {
  1331. color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1332. }
  1333. mujs-body table .frame[data-good] mujs-a:hover, mujs-body table .frame[data-author] mujs-a:hover {
  1334. color: hsl(81, 56%, 43%);
  1335. }
  1336. mujs-body table .frame[data-good] .mujs-list, mujs-body table .frame[data-author] .mujs-list {
  1337. color: hsl(0, 0%, 100%);
  1338. }
  1339. mujs-body table .frame[data-good] mu-jsbtn, mujs-body table .frame[data-author] mu-jsbtn {
  1340. color: hsl(215, 47%, 24%);
  1341. background: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1342. border-color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1343. }
  1344. mujs-body table .frame[data-good] mu-jsbtn:hover, mujs-body table .frame[data-author] mu-jsbtn:hover {
  1345. background: hsl(81, 56%, 65%);
  1346. border-color: hsl(81, 56%, 65%);
  1347. }
  1348. mujs-body table .frame.translated:not([data-good], [data-author]) mujs-a {
  1349. color: hsl(249, 56%, 65%);
  1350. }
  1351. mujs-body table .frame.translated:not([data-good], [data-author]) mujs-a:hover {
  1352. color: hsl(249, 56%, 85%);
  1353. }
  1354. mujs-body table .frame.translated:not([data-good], [data-author]) mu-jsbtn {
  1355. color: hsl(215, 47%, 85%);
  1356. background: hsl(249, 56%, 65%);
  1357. border-color: hsl(249, 56%, 65%);
  1358. }
  1359. mujs-body table .frame.translated:not([data-good], [data-author]) mu-jsbtn:hover {
  1360. background: hsl(249, 56%, 65%);
  1361. border-color: hsl(249, 56%, 65%);
  1362. }
  1363. mujs-body table .frame .mujs-ratings[data-el=good] {
  1364. border-color: hsl(120, 50%, 40%);
  1365. background-color: hsla(120, 50%, 40%, 0.102);
  1366. color: hsl(120, 100%, 60%);
  1367. }
  1368. mujs-body table .frame .mujs-ratings[data-el=ok] {
  1369. border-color: hsl(60, 100%, 30%);
  1370. background-color: hsla(60, 100%, 30%, 0.102);
  1371. color: hsl(60, 100%, 50%);
  1372. }
  1373. mujs-body table .frame .mujs-ratings[data-el=bad] {
  1374. border-color: hsl(0, 100%, 30%);
  1375. background-color: hsla(0, 50%, 40%, 0.102);
  1376. color: hsl(0, 100%, 50%);
  1377. }
  1378. mujs-body table .frame svg {
  1379. width: 12px;
  1380. height: 12px;
  1381. fill: currentColor;
  1382. background: transparent;
  1383. }
  1384. mujs-body table .frame > td:not(.mujs-name) {
  1385. text-align: center;
  1386. }
  1387. mujs-body table .frame > .mujs-name > mujs-a {
  1388. width: -webkit-fit-content;
  1389. width: -moz-fit-content;
  1390. width: fit-content;
  1391. }
  1392. mujs-body table .frame > .mujs-name mu-jsbtn,
  1393. mujs-body table .frame > .mujs-name mu-js {
  1394. height: -webkit-fit-content;
  1395. height: -moz-fit-content;
  1396. height: fit-content;
  1397. }
  1398. mujs-body table .frame > .mujs-name > mu-jsbtn {
  1399. margin: auto;
  1400. }
  1401. mujs-body table .frame > .mujs-name > mujs-column > mu-jsbtn {
  1402. padding: 0px 7px;
  1403. }
  1404. @media screen and (max-width: 800px) {
  1405. mujs-body table .frame > .mujs-name > mujs-column > mu-jsbtn {
  1406. width: 100%;
  1407. }
  1408. }
  1409. mujs-body table .frame > .mujs-uframe > mujs-a {
  1410. font-size: 16px;
  1411. font-weight: 500;
  1412. padding-left: 0.5rem;
  1413. padding-right: 0.5rem;
  1414. }
  1415. mujs-body table .frame [data-el=more-info] > mujs-row {
  1416. gap: 0.25em;
  1417. }
  1418. mujs-body table .frame [data-el=matches] {
  1419. gap: 0.25em;
  1420. max-width: 40em;
  1421. }
  1422. mujs-body table .frame [data-el=matches] .mujs-grants {
  1423. display: inline-flex;
  1424. flex-flow: row wrap;
  1425. overflow: auto;
  1426. overflow-wrap: break-word;
  1427. text-overflow: ellipsis;
  1428. white-space: nowrap;
  1429. width: -webkit-fit-content;
  1430. width: -moz-fit-content;
  1431. width: fit-content;
  1432. max-height: 5em;
  1433. gap: 0.2em;
  1434. }
  1435. mujs-body table .frame [data-el=matches] .mujs-grants > mujs-a {
  1436. display: inline;
  1437. }
  1438. mujs-body table .frame [data-el=matches] .mujs-grants > mujs-a:not([data-command]) {
  1439. cursor: default !important;
  1440. color: var(--mujs-txt-color, hsl(0, 0%, 100%));
  1441. }
  1442. mujs-body table .frame [data-el=matches] .mujs-grants > mujs-a::after {
  1443. content: ", ";
  1444. color: var(--mujs-txt-color, hsl(0, 0%, 100%));
  1445. }
  1446. mujs-body table .frame [data-el=matches] .mujs-grants > mujs-a:last-child::after {
  1447. content: "";
  1448. }
  1449. @media screen and (max-width: 800px) {
  1450. mujs-body table .frame [data-el=matches] {
  1451. align-self: center;
  1452. width: 30em !important;
  1453. }
  1454. }
  1455. mujs-body table .frame [data-name=license] {
  1456. text-overflow: ellipsis;
  1457. overflow: hidden;
  1458. white-space: nowrap;
  1459. width: -webkit-fit-content;
  1460. width: -moz-fit-content;
  1461. width: fit-content;
  1462. }
  1463. @media screen and (max-width: 800px) {
  1464. mujs-body table .frame [data-name=license] {
  1465. width: 100% !important;
  1466. width: -moz-available !important;
  1467. width: -webkit-fill-available !important;
  1468. }
  1469. }
  1470. @media screen and (max-width: 1150px) {
  1471. .mujs-cfg {
  1472. margin: 0px auto 1rem auto !important;
  1473. }
  1474. }
  1475. .mujs-cfg {
  1476. height: fit-content;
  1477. height: -moz-fit-content;
  1478. height: -webkit-fit-content;
  1479. }
  1480. .mujs-cfg mujs-section {
  1481. border-radius: 16px;
  1482. padding: 0.5em;
  1483. }
  1484. .mujs-cfg mujs-section:nth-child(even) {
  1485. background: var(--mujs-even-row, hsl(222, 14%, 18%)) !important;
  1486. }
  1487. .mujs-cfg mujs-section:nth-child(even) input,
  1488. .mujs-cfg mujs-section:nth-child(even) select {
  1489. background: var(--mujs-odd-row, hsl(222, 14%, 33%));
  1490. }
  1491. .mujs-cfg mujs-section:nth-child(even) select option {
  1492. background: var(--mujs-odd-row, hsl(222, 14%, 33%));
  1493. }
  1494. .mujs-cfg mujs-section:nth-child(even) select option:hover {
  1495. background: var(--mujs-even-row, hsl(222, 14%, 18%)) !important;
  1496. }
  1497. .mujs-cfg mujs-section:nth-child(odd) {
  1498. background: var(--mujs-odd-row, hsl(222, 14%, 33%)) !important;
  1499. }
  1500. .mujs-cfg mujs-section:nth-child(odd) input,
  1501. .mujs-cfg mujs-section:nth-child(odd) select {
  1502. background: var(--mujs-even-row, hsl(222, 14%, 18%));
  1503. }
  1504. .mujs-cfg mujs-section:nth-child(odd) select option {
  1505. background: var(--mujs-even-row, hsl(222, 14%, 18%));
  1506. }
  1507. .mujs-cfg mujs-section:nth-child(odd) select option:hover {
  1508. background: var(--mujs-odd-row, hsl(222, 14%, 33%)) !important;
  1509. }
  1510. .mujs-cfg mujs-section[data-name=theme] .sub-section {
  1511. border-radius: 4px;
  1512. }
  1513. .mujs-cfg mujs-section[data-name=theme] .sub-section:nth-child(even) {
  1514. background: var(--mujs-even-row, hsl(222, 14%, 18%));
  1515. }
  1516. .mujs-cfg mujs-section[data-name=theme] .sub-section:nth-child(odd) {
  1517. background: var(--mujs-odd-row, hsl(222, 14%, 33%));
  1518. }
  1519. .mujs-cfg mujs-section[data-name=theme] input,
  1520. .mujs-cfg mujs-section[data-name=theme] select {
  1521. background: inherit;
  1522. }
  1523. .mujs-cfg mujs-section[data-name=theme] select option {
  1524. background: inherit;
  1525. }
  1526. .mujs-cfg mujs-section[data-name=theme] select option:hover {
  1527. background: var(--mujs-even-row, hsl(222, 14%, 18%)) !important;
  1528. }
  1529. .mujs-cfg mujs-section[data-name=exp], .mujs-cfg mujs-section[data-name=blacklist] {
  1530. display: flex;
  1531. justify-content: space-between;
  1532. flex-direction: column;
  1533. gap: 0.25em;
  1534. }
  1535. .mujs-cfg mujs-section[data-name=exp] > mujs-btn, .mujs-cfg mujs-section[data-name=blacklist] > mujs-btn {
  1536. width: 100%;
  1537. width: -moz-available;
  1538. width: -webkit-fill-available;
  1539. }
  1540. .mujs-cfg mujs-section[data-name=exp] > mujs-btn:hover, .mujs-cfg mujs-section[data-name=blacklist] > mujs-btn:hover {
  1541. background: var(--mujs-even-row, hsl(222, 14%, 18%)) !important;
  1542. }
  1543. .mujs-cfg mujs-section input[type=text]::-webkit-input-placeholder {
  1544. color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1545. }
  1546. .mujs-cfg mujs-section input[type=text]::-moz-placeholder {
  1547. color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1548. }
  1549. .mujs-cfg mujs-section input[type=text]:-ms-input-placeholder {
  1550. color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1551. }
  1552. .mujs-cfg mujs-section input[type=text]::-ms-input-placeholder {
  1553. color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1554. }
  1555. .mujs-cfg mujs-section input[type=text]::placeholder {
  1556. color: var(--mujs-placeholder, hsl(81, 56%, 54%));
  1557. }
  1558. .mujs-cfg mujs-section > label:not([data-blacklist]) {
  1559. display: flex;
  1560. justify-content: space-between;
  1561. }
  1562. .mujs-cfg mujs-section > label[data-blacklist] {
  1563. display: grid;
  1564. grid-auto-flow: column;
  1565. }
  1566. .mujs-cfg mujs-section > label[data-blacklist]:not(.new-list) {
  1567. grid-template-columns: repeat(2, 1fr);
  1568. }
  1569. .mujs-cfg mujs-section > label.new-list {
  1570. order: 999999999999;
  1571. }
  1572. .mujs-cfg mujs-section > label.new-list mujs-add {
  1573. font-size: 20px;
  1574. }
  1575. .mujs-cfg mujs-section > label input:not([type=checkbox]) {
  1576. font-size: 14px;
  1577. position: relative;
  1578. border-radius: 4px;
  1579. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1580. }
  1581. .mujs-cfg mujs-section select,
  1582. .mujs-cfg mujs-section select option {
  1583. color: var(--mujs-txt-color, hsl(0, 0%, 100%));
  1584. border: 1px solid transparent;
  1585. list-style: none;
  1586. outline-style: none;
  1587. pointer-events: auto;
  1588. }
  1589. .mujs-cfg mujs-section select {
  1590. text-align: center;
  1591. border-radius: 4px;
  1592. }
  1593. .mujs-cfg mujs-section > *.sub-section {
  1594. padding: 0.2em;
  1595. }
  1596. .mujs-cfg mujs-section > *.sub-section[data-engine] {
  1597. flex-wrap: wrap;
  1598. }
  1599. .mujs-cfg mujs-section > *.sub-section[data-engine] input {
  1600. width: 100%;
  1601. width: -moz-available;
  1602. width: -webkit-fill-available;
  1603. }
  1604. .mujs-cfg mujs-section > *.sub-section input[type=text] {
  1605. margin: 0.2em 0px;
  1606. }
  1607. .mujs-cfg .mujs-inlab {
  1608. position: relative;
  1609. width: 38px;
  1610. }
  1611. .mujs-cfg .mujs-inlab input[type=checkbox] {
  1612. display: none;
  1613. }
  1614. .mujs-cfg .mujs-inlab input[type=checkbox]:checked + label {
  1615. margin-left: 0;
  1616. background: var(--mujs-chck-color, hsla(0, 0%, 100%, 0.568));
  1617. }
  1618. .mujs-cfg .mujs-inlab input[type=checkbox]:checked + label:before {
  1619. right: 0px;
  1620. }
  1621. .mujs-cfg .mujs-inlab input[type=checkbox][data-name=greasyfork]:checked + label {
  1622. background: var(--mujs-gf-color, hsl(204, 100%, 40%));
  1623. }
  1624. .mujs-cfg .mujs-inlab input[type=checkbox][data-name=sleazyfork]:checked + label {
  1625. background: var(--mujs-sf-color, hsl(12, 86%, 50%));
  1626. }
  1627. .mujs-cfg .mujs-inlab input[type=checkbox][data-name=openuserjs]:checked + label {
  1628. background: var(--mujs-chck-open, hsla(12, 86%, 50%, 0.568));
  1629. }
  1630. .mujs-cfg .mujs-inlab input[type=checkbox][data-name=github]:checked + label {
  1631. background: var(--mujs-chck-git, hsla(213, 13%, 16%, 0.568));
  1632. }
  1633. .mujs-cfg .mujs-inlab label {
  1634. padding: 0;
  1635. display: block;
  1636. overflow: hidden;
  1637. height: 16px;
  1638. border-radius: 20px;
  1639. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1640. }
  1641. .mujs-cfg .mujs-inlab label:before {
  1642. content: "";
  1643. display: block;
  1644. width: 20px;
  1645. height: 20px;
  1646. margin: -2px;
  1647. background: var(--mujs-txt-color, hsl(0, 0%, 100%));
  1648. position: absolute;
  1649. top: 0;
  1650. right: 20px;
  1651. border-radius: 20px;
  1652. }
  1653. .mujs-cfg .mujs-sty-flex mujs-btn {
  1654. margin: auto;
  1655. }
  1656. .mujs-cfg .mujs-sty-flex mujs-btn[data-command=reset] {
  1657. background: var(--mujs-sf-btn-color, hsl(12, 86%, 50%));
  1658. border-color: var(--mujs-sf-btn-color, hsl(12, 86%, 50%));
  1659. }
  1660. .mujs-cfg .mujs-sty-flex mujs-btn[data-command=reset]:hover {
  1661. background: var(--mujs-sf-txt-color, hsl(12, 79%, 55%));
  1662. border-color: var(--mujs-sf-txt-color, hsl(12, 79%, 55%));
  1663. }
  1664. .mujs-cfg .mujs-sty-flex mujs-btn[data-command=save] {
  1665. background: var(--mujs-gf-color, hsl(204, 100%, 40%));
  1666. border-color: var(--mujs-gf-color, hsl(204, 100%, 40%));
  1667. }
  1668. .mujs-cfg .mujs-sty-flex mujs-btn[data-command=save]:hover {
  1669. background: var(--mujs-gf-btn-color, hsl(211, 87%, 56%));
  1670. border-color: var(--mujs-gf-btn-color, hsl(211, 87%, 56%));
  1671. }
  1672. .mujs-cfg:not(.webext-page) {
  1673. margin: 1rem 25rem;
  1674. }
  1675. @media screen and (max-height: 720px) {
  1676. .mujs-cfg:not(.webext-page) {
  1677. height: 100%;
  1678. height: -moz-available;
  1679. height: -webkit-fill-available;
  1680. width: 100%;
  1681. width: -moz-available;
  1682. width: -webkit-fill-available;
  1683. overflow-x: auto;
  1684. padding: 0.5em;
  1685. }
  1686. }
  1687. mujs-a {
  1688. display: inline-block;
  1689. }
  1690. .mujs-name {
  1691. display: flex;
  1692. flex-flow: column wrap;
  1693. gap: 0.5em;
  1694. }
  1695. .mujs-name span {
  1696. font-size: 0.8em !important;
  1697. }
  1698. mujs-btn {
  1699. font-style: normal;
  1700. font-weight: 500;
  1701. font-variant: normal;
  1702. text-transform: none;
  1703. text-rendering: auto;
  1704. text-align: center;
  1705. border: 1px solid var(--mujs-txt-color, hsl(0, 0%, 100%));
  1706. font-size: 16px;
  1707. border-radius: 4px;
  1708. line-height: 1;
  1709. padding: 6px 15px;
  1710. }
  1711. mujs-btn svg {
  1712. width: 14px;
  1713. height: 14px;
  1714. fill: var(--mujs-txt-color, hsl(0, 0%, 100%));
  1715. }
  1716. mu-jsbtn {
  1717. font-size: 14px;
  1718. border-radius: 4px;
  1719. font-style: normal;
  1720. padding: 7px 15%;
  1721. font-weight: 400;
  1722. font-variant: normal;
  1723. line-height: normal;
  1724. display: block;
  1725. text-align: center;
  1726. }
  1727. mujs-a,
  1728. mu-jsbtn,
  1729. .mujs-pointer,
  1730. .mujs-cfg mujs-section *:not(input[type=text], input[type=number], [data-theme], [data-blacklist]),
  1731. .mainbtn,
  1732. .mainframe,
  1733. mujs-btn {
  1734. cursor: pointer !important;
  1735. }`;
  1736. /******************************************************************************/
  1737. // #region Console
  1738. const con = {
  1739. title: '[%cMagic Userscript+%c]',
  1740. color: 'color: rgb(29, 155, 240);',
  1741. dbg(...msg) {
  1742. const dt = new Date();
  1743. console.debug(
  1744. `${con.title} %cDBG`,
  1745. con.color,
  1746. '',
  1747. 'color: rgb(255, 212, 0);',
  1748. `[${dt.getHours()}:${('0' + dt.getMinutes()).slice(-2)}:${('0' + dt.getSeconds()).slice(-2)}]`,
  1749. ...msg
  1750. );
  1751. },
  1752. err(...msg) {
  1753. console.error(
  1754. `${con.title} %cERROR`,
  1755. con.color,
  1756. '',
  1757. 'color: rgb(249, 24, 128);',
  1758. ...msg
  1759. );
  1760. const a = typeof alert !== 'undefined' && alert;
  1761. for (const ex of msg) {
  1762. if (typeof ex === 'object' && 'cause' in ex && a) {
  1763. a(`[Magic Userscript+] (${ex.cause}) ${ex.message}`);
  1764. }
  1765. }
  1766. },
  1767. info(...msg) {
  1768. console.info(
  1769. `${con.title} %cINF`,
  1770. con.color,
  1771. '',
  1772. 'color: rgb(0, 186, 124);',
  1773. ...msg
  1774. );
  1775. },
  1776. log(...msg) {
  1777. console.log(
  1778. `${con.title} %cLOG`,
  1779. con.color,
  1780. '',
  1781. 'color: rgb(219, 160, 73);',
  1782. ...msg
  1783. );
  1784. }
  1785. };
  1786. const { err, info } = con;
  1787. // #endregion
  1788. /**
  1789. * @type { import("../typings/types.d.ts").config }
  1790. */
  1791. let cfg;
  1792. const BLANK_FN = function () {};
  1793. const BLANK_ASYNC_FN = async function () {};
  1794. const BLANK_PAGE = 'about:blank';
  1795. /**
  1796. * @param {string} hn
  1797. */
  1798. const normalizedHostname = (hn) => hn.replace(/^www\./, '');
  1799. /**
  1800. * @param {string} txt
  1801. */
  1802. const formatURL = (txt) =>
  1803. txt
  1804. .split('.')
  1805. .splice(-2)
  1806. .join('.')
  1807. .replace(/\/|https:/g, '');
  1808. /**
  1809. * @param {string} str
  1810. */
  1811. const getHostname = (str) => formatURL(normalizedHostname(str));
  1812. /**
  1813. * @type {URL | undefined}
  1814. */
  1815. let url;
  1816. try {
  1817. url = new URL(window.location.href ?? BLANK_PAGE);
  1818. } catch {
  1819. /* empty */
  1820. }
  1821. // #region Validators
  1822. const objToStr = (obj) => Object.prototype.toString.call(obj);
  1823. const isRegExp = (obj) => /RegExp/.test(objToStr(obj));
  1824. const isElem = (obj) => /Element/.test(objToStr(obj));
  1825. const isHTML = (obj) => /object HTML/.test(objToStr(obj));
  1826. const isObj = (obj) => /Object/.test(objToStr(obj));
  1827. const isFN = (obj) => /Function/.test(objToStr(obj));
  1828. const isUserCSS = (str) => /\.user\.css$/.test(str);
  1829. const isUserJS = (str) => /\.user\.js$/.test(str);
  1830. /**
  1831. * @type { import("../typings/types.d.ts").isNull }
  1832. */
  1833. const isNull = (obj) => {
  1834. return Object.is(obj, null) || Object.is(obj, undefined);
  1835. };
  1836. /**
  1837. * @type { import("../typings/types.d.ts").isBlank }
  1838. */
  1839. const isBlank = (obj) => {
  1840. return (
  1841. (typeof obj === 'string' && Object.is(obj.trim(), '')) ||
  1842. ((obj instanceof Set || obj instanceof Map) && Object.is(obj.size, 0)) ||
  1843. (Array.isArray(obj) && Object.is(obj.length, 0)) ||
  1844. (isObj(obj) && Object.is(Object.keys(obj).length, 0))
  1845. );
  1846. };
  1847. /**
  1848. * @type { import("../typings/types.d.ts").isEmpty }
  1849. */
  1850. const isEmpty = (obj) => {
  1851. return isNull(obj) || isBlank(obj);
  1852. };
  1853. // #endregion
  1854. // #region Globals
  1855. /**
  1856. * https://github.com/zloirock/core-js/blob/master/packages/core-js/internals/global-this.js
  1857. * @returns {typeof globalThis}
  1858. */
  1859. function globalWin() {
  1860. const check = function (it) {
  1861. return it && it.Math === Math && it;
  1862. };
  1863. return (
  1864. check(typeof globalThis == 'object' && globalThis) ||
  1865. check(typeof window == 'object' && window) ||
  1866. check(typeof self == 'object' && self) ||
  1867. check(typeof this == 'object' && this) ||
  1868. (function () {
  1869. return this;
  1870. })() ||
  1871. Function('return this')()
  1872. );
  1873. }
  1874. class Safe {
  1875. /** @type { import("../typings/UserJS.d.ts").safeHandles } */
  1876. _safe;
  1877. create() {
  1878. try {
  1879. const g = globalWin();
  1880. /** @type { import("../typings/UserJS.d.ts").safeHandles } */
  1881. const safe = {
  1882. XMLHttpRequest: g.XMLHttpRequest,
  1883. CustomEvent: g.CustomEvent,
  1884. createElement: g.document.createElement.bind(g.document),
  1885. createElementNS: g.document.createElementNS.bind(g.document),
  1886. createTextNode: g.document.createTextNode.bind(g.document),
  1887. setTimeout: g.setTimeout,
  1888. clearTimeout: g.clearTimeout,
  1889. navigator: g.navigator,
  1890. scheduler: {
  1891. postTask(callback, options) {
  1892. if ('scheduler' in g && 'postTask' in g.scheduler) {
  1893. return g.scheduler.postTask(callback, options);
  1894. }
  1895. options = Object.assign({}, options);
  1896. if (options.delay === undefined) options.delay = 0;
  1897. options.delay = Number(options.delay);
  1898. if (options.delay < 0) {
  1899. return Promise.reject(new TypeError('"delay" must be a positive number.'));
  1900. }
  1901. return new Promise((resolve) => {
  1902. g.setTimeout(() => {
  1903. resolve(callback());
  1904. }, options.delay);
  1905. });
  1906. },
  1907. yield() {
  1908. if ('scheduler' in g && 'yield' in g.scheduler) {
  1909. scheduler.yield();
  1910. return g.scheduler.yield();
  1911. }
  1912. return new Promise((resolve) => {
  1913. g.setTimeout(resolve, 0);
  1914. });
  1915. }
  1916. },
  1917. groupBy(arr, callback) {
  1918. if (isFN(Object.groupBy)) {
  1919. return Object.groupBy(arr, callback);
  1920. }
  1921. /** [Object.groupBy polyfill](https://gist.github.com/gtrabanco/7c97bd41aa74af974fa935bfb5044b6e) */
  1922. return arr.reduce((acc = {}, ...args) => {
  1923. const key = callback(...args);
  1924. acc[key] ??= [];
  1925. acc[key].push(args[0]);
  1926. return acc;
  1927. }, {});
  1928. }
  1929. };
  1930. for (const [k, v] of Object.entries(safe)) {
  1931. if (/scheduler|navigator/.test(k) || isFN(v)) continue;
  1932. throw new Error(`Safe handles "${k}" returned "${v}"`, { cause: 'safeSelf' });
  1933. }
  1934. this._safe = safe;
  1935. } catch (e) {
  1936. err(e);
  1937. this._safe = null;
  1938. }
  1939. return this._safe;
  1940. }
  1941. get _self() {
  1942. return this._safe ?? this.create();
  1943. }
  1944. set _self(obj) {
  1945. this._safe = obj;
  1946. }
  1947. }
  1948. const { _self } = new Safe();
  1949. // #endregion
  1950. // #region Constants
  1951. /**
  1952. * @type { import("../typings/types.d.ts").cfgBase }
  1953. */
  1954. const cfgBase = [];
  1955. const cfgSec = new Set();
  1956. /** Lets highlight me :) */
  1957. const authorID = 166061;
  1958. /**
  1959. * Some UserJS I personally enjoy - `https://greasyfork.org/scripts/{id}`
  1960. */
  1961. const goodUserJS = [
  1962. 33005,
  1963. 394820,
  1964. 438684,
  1965. 4870,
  1966. 394420,
  1967. 25068,
  1968. 483444,
  1969. 1682,
  1970. 22587,
  1971. 789,
  1972. 28497,
  1973. 386908,
  1974. 24204,
  1975. 404443,
  1976. 4336,
  1977. 368183,
  1978. 393396,
  1979. 473830,
  1980. 12179,
  1981. 423001,
  1982. 376510,
  1983. 23840,
  1984. 40525,
  1985. 6456,
  1986. 'https://openuserjs.org/install/Patabugen/Always_Remember_Me.user.js',
  1987. 'https://openuserjs.org/install/nokeya/Direct_links_out.user.js',
  1988. 'https://github.com/jijirae/y2monkey/raw/main/y2monkey.user.js',
  1989. 'https://github.com/jijirae/r2monkey/raw/main/r2monkey.user.js',
  1990. 'https://github.com/TagoDR/MangaOnlineViewer/raw/master/Manga_OnlineViewer.user.js',
  1991. 'https://github.com/j###s2099/konami-command/raw/master/INSTALL-USER-SCRIPT.user.js',
  1992. 'https://github.com/TagoDR/MangaOnlineViewer/raw/master/dist/Manga_OnlineViewer_Adult.user.js'
  1993. ];
  1994. /** Remove UserJS from banned accounts */
  1995. const badUserJS = [478597];
  1996. /** Unsupport host for search engines */
  1997. const engineUnsupported = {
  1998. greasyfork: ['pornhub.com'],
  1999. sleazyfork: ['pornhub.com'],
  2000. openuserjs: [],
  2001. github: []
  2002. };
  2003. const isMobile = (() => {
  2004. if (userjs.isMobile !== undefined) {
  2005. return userjs.isMobile;
  2006. }
  2007. try {
  2008. const { navigator } = _self;
  2009. if (navigator) {
  2010. const { userAgent, userAgentData } = navigator;
  2011. const { platform, mobile } = userAgentData ? Object(userAgentData) : {};
  2012. userjs.isMobile =
  2013. /Mobile|Tablet/.test(userAgent ? String(userAgent) : '') ||
  2014. Boolean(mobile) ||
  2015. /Android|Apple/.test(platform ? String(platform) : '');
  2016. } else {
  2017. userjs.isMobile = false;
  2018. }
  2019. } catch (ex) {
  2020. userjs.isMobile = false;
  2021. ex.cause = 'getUAData';
  2022. err(ex);
  2023. }
  2024. return userjs.isMobile;
  2025. })();
  2026. const isGM = typeof GM !== 'undefined';
  2027. const builtinList = {
  2028. local: /localhost|router|###|(\d+\.){3}\d+/,
  2029. finance:
  2030. /school|pay|bank|money|cart|checkout|authorize|bill|wallet|venmo|zalo|skrill|bluesnap|coin|crypto|currancy|insurance|finance/,
  2031. social: /login|join|signin|signup|sign-up|password|reset|password_reset/,
  2032. unsupported: {
  2033. host: 'fakku.net',
  2034. pathname: '/hentai/.+/read/page/.+'
  2035. }
  2036. };
  2037. // #endregion
  2038. // #region DEFAULT_CONFIG
  2039. /**
  2040. * @type { import("../typings/types.d.ts").config }
  2041. */
  2042. const DEFAULT_CONFIG = {
  2043. autofetch: true,
  2044. autoinject: true,
  2045. autoSort: 'daily_installs',
  2046. clearTabCache: true,
  2047. cache: true,
  2048. autoexpand: false,
  2049. filterlang: false,
  2050. sleazyredirect: false,
  2051. time: 10000,
  2052. blacklist: ['userjs-local', 'userjs-finance', 'userjs-social', 'userjs-unsupported'],
  2053. preview: {
  2054. code: false,
  2055. metadata: false
  2056. },
  2057. engines: [
  2058. {
  2059. enabled: true,
  2060. name: 'greasyfork',
  2061. query: encodeURIComponent('https://greasyfork.org/scripts/by-site/{host}.json?language=all')
  2062. },
  2063. {
  2064. enabled: false,
  2065. name: 'sleazyfork',
  2066. query: encodeURIComponent('https://sleazyfork.org/scripts/by-site/{host}.json?language=all')
  2067. },
  2068. {
  2069. enabled: false,
  2070. name: 'openuserjs',
  2071. query: encodeURIComponent('https://openuserjs.org/?q={host}')
  2072. },
  2073. {
  2074. enabled: false,
  2075. name: 'github',
  2076. token: '',
  2077. query: encodeURIComponent(
  2078. 'https://api.github.com/search/repositories?q=topic:{domain}+topic:userscript'
  2079. )
  2080. }
  2081. ],
  2082. theme: {
  2083. 'even-row': '',
  2084. 'odd-row': '',
  2085. 'even-err': '',
  2086. 'odd-err': '',
  2087. 'background-color': '',
  2088. 'gf-color': '',
  2089. 'sf-color': '',
  2090. 'border-b-color': '',
  2091. 'gf-btn-color': '',
  2092. 'sf-btn-color': '',
  2093. 'sf-txt-color': '',
  2094. 'txt-color': '',
  2095. 'chck-color': '',
  2096. 'chck-gf': '',
  2097. 'chck-git': '',
  2098. 'chck-open': '',
  2099. placeholder: '',
  2100. 'position-top': '',
  2101. 'position-bottom': '',
  2102. 'position-left': '',
  2103. 'position-right': '',
  2104. 'font-family': ''
  2105. },
  2106. recommend: {
  2107. author: true,
  2108. others: true
  2109. },
  2110. filters: {
  2111. ASCII: {
  2112. enabled: false,
  2113. name: 'Non-ASCII',
  2114. regExp: '[^\\x00-\\x7F\\s]+'
  2115. },
  2116. Latin: {
  2117. enabled: false,
  2118. name: 'Non-Latin',
  2119. regExp: '[^\\u0000-\\u024F\\u2000-\\u214F\\s]+'
  2120. },
  2121. Games: {
  2122. enabled: false,
  2123. name: 'Games',
  2124. flag: 'iu',
  2125. regExp:
  2126. 'Aimbot|AntiGame|Agar|agar\\.io|alis\\.io|angel\\.io|ExtencionRipXChetoMalo|AposBot|DFxLite|ZTx-Lite|AposFeedingBot|AposLoader|Balz|Blah Blah|Orc Clan Script|Astro\\s*Empires|^\\s*Attack|^\\s*Battle|BiteFight|Blood\\s*Wars|Bloble|Bonk|Bots|Bots4|Brawler|\\bBvS\\b|Business\\s*Tycoon|Castle\\s*Age|City\\s*Ville|chopcoin\\.io|Comunio|Conquer\\s*Club|CosmoPulse|cursors\\.io|Dark\\s*Orbit|Dead\\s*Frontier|Diep\\.io|\\bDOA\\b|doblons\\.io|DotD|Dossergame|Dragons\\s*of\\s*Atlantis|driftin\\.io|Dugout|\\bDS[a-z]+\\n|elites\\.io|Empire\\s*Board|eRep(ublik)?|Epicmafia|Epic.*War|ExoPlanet|Falcon Tools|Feuerwache|Farming|FarmVille|Fightinfo|Frontier\\s*Ville|Ghost\\s*Trapper|Gladiatus|Goalline|Gondal|gota\\.io|Grepolis|Hobopolis|\\bhwm(\\b|_)|Ikariam|\\bIT2\\b|Jellyneo|Kapi\\s*Hospital|Kings\\s*Age|Kingdoms?\\s*of|knastv(o|oe)gel|Knight\\s*Fight|\\b(Power)?KoC(Atta?ck)?\\b|\\bKOL\\b|Kongregate|Krunker|Last\\s*Emperor|Legends?\\s*of|Light\\s*Rising|lite\\.ext\\.io|Lockerz|\\bLoU\\b|Mafia\\s*(Wars|Mofo)|Menelgame|Mob\\s*Wars|Mouse\\s*Hunt|Molehill\\s*Empire|MooMoo|MyFreeFarm|narwhale\\.io|Neopets|NeoQuest|Nemexia|\\bOGame\\b|Ogar(io)?|Pardus|Pennergame|Pigskin\\s*Empire|PlayerScripts|pokeradar\\.io|Popmundo|Po?we?r\\s*(Bot|Tools)|PsicoTSI|Ravenwood|Schulterglatze|Skribbl|slither\\.io|slitherplus\\.io|slitheriogameplay|SpaceWars|splix\\.io|Survivio|\\bSW_[a-z]+\\n|\\bSnP\\b|The\\s*Crims|The\\s*West|torto\\.io|Travian|Treasure\\s*Isl(and|e)|Tribal\\s*Wars|TW.?PRO|Vampire\\s*Wars|vertix\\.io|War\\s*of\\s*Ninja|World\\s*of\\s*####s|West\\s*Wars|wings\\.io|\\bWoD\\b|World\\s*of\\s*Dungeons|wtf\\s*battles|Wurzelimperium|Yohoho|Zombs'
  2127. },
  2128. SocialNetworks: {
  2129. enabled: false,
  2130. name: 'Social Networks',
  2131. flag: 'iu',
  2132. regExp:
  2133. 'Face\\s*book|Google(\\+| Plus)|\\bHabbo|Kaskus|\\bLepra|Leprosorium|MySpace|meinVZ|odnoklassniki|Одноклассники|Orkut|sch(ue|ü)ler(VZ|\\.cc)?|studiVZ|Unfriend|Valenth|VK|vkontakte|ВКонтакте|Qzone|Twitter|TweetDeck'
  2134. },
  2135. Clutter: {
  2136. enabled: false,
  2137. name: 'Clutter',
  2138. flag: 'iu',
  2139. regExp:
  2140. "^\\s*(.{1,3})\\1+\\n|^\\s*(.+?)\\n+\\2\\n*$|^\\s*.{1,5}\\n|do\\s*n('|o)?t (install|download)|nicht installieren|(just )?(\\ban? |\\b)test(ing|s|\\d|\\b)|^\\s*.{0,4}test.{0,4}\\n|\\ntest(ing)?\\s*|^\\s*(\\{@|Smolka|Hacks)|\\[\\d{4,5}\\]|free\\s*download|theme|(night|dark) ?(mode)?"
  2141. }
  2142. }
  2143. };
  2144. // #endregion
  2145. // #region i18n
  2146. /** @type {Map<string, string>} */
  2147. const i18nMap = new Map();
  2148. class i18nHandler {
  2149. constructor() {
  2150. for (const [k, v] of Object.entries(translations)) {
  2151. if (!i18nMap.has(k)) i18nMap.set(k, v);
  2152. }
  2153. }
  2154. /**
  2155. * @param {string | Date | number} str
  2156. */
  2157. toDate(str = '') {
  2158. const { navigator } = _self;
  2159. return new Intl.DateTimeFormat(navigator.language).format(
  2160. typeof str === 'string' ? new Date(str) : str
  2161. );
  2162. }
  2163. /**
  2164. * @param {number | bigint} number
  2165. */
  2166. toNumber(number) {
  2167. const { navigator } = _self;
  2168. return new Intl.NumberFormat(navigator.language).format(number);
  2169. }
  2170. /**
  2171. * @type { import("../typings/UserJS.d.ts").i18n$ }
  2172. */
  2173. i18n$(key) {
  2174. const { navigator } = _self;
  2175. const current = navigator.language.split('-')[0] ?? 'en';
  2176. try {
  2177. return i18nMap.get(current)?.[key] ?? 'Invalid Key'
  2178. } catch (e) {
  2179. err(e);
  2180. return 'error';
  2181. }
  2182. }
  2183. get current() {
  2184. const { navigator } = _self;
  2185. return navigator.language.split('-')[0] ?? 'en';
  2186. }
  2187. }
  2188. const language = new i18nHandler();
  2189. const { i18n$ } = language;
  2190. // #endregion
  2191. // #region Utilities
  2192. const union = (...arr) => [...new Set(arr.flat())];
  2193. /**
  2194. * @param {string} str
  2195. */
  2196. const decode = (str) => {
  2197. try {
  2198. if (decodeURI(str) !== decodeURIComponent(str)) {
  2199. return decode(decodeURIComponent(str));
  2200. }
  2201. } catch (ex) {
  2202. err(ex);
  2203. }
  2204. return str;
  2205. };
  2206. /**
  2207. * @type { import("../typings/types.d.ts").normalizeTarget }
  2208. */
  2209. const normalizeTarget = (target, toQuery = true, root) => {
  2210. if (Object.is(target, null) || Object.is(target, undefined)) {
  2211. return [];
  2212. }
  2213. if (Array.isArray(target)) {
  2214. return target;
  2215. }
  2216. if (typeof target === 'string') {
  2217. return toQuery ? Array.from((root || document).querySelectorAll(target)) : Array.of(target);
  2218. }
  2219. if (/object HTML/.test(Object.prototype.toString.call(target))) {
  2220. return Array.of(target);
  2221. }
  2222. return Array.from(target);
  2223. };
  2224. /**
  2225. * @type { import("../typings/types.d.ts").qs }
  2226. */
  2227. const qs = (selector, root) => {
  2228. try {
  2229. return (root || document).querySelector(selector);
  2230. } catch (ex) {
  2231. err(ex);
  2232. }
  2233. return null;
  2234. };
  2235. /**
  2236. * @type { import("../typings/types.d.ts").qsA }
  2237. */
  2238. const qsA = (selectors, root) => {
  2239. try {
  2240. return (root || document).querySelectorAll(selectors);
  2241. } catch (ex) {
  2242. err(ex);
  2243. }
  2244. return [];
  2245. };
  2246. /**
  2247. * @type { import("../typings/types.d.ts").ael }
  2248. */
  2249. const ael = (el, type, listener, options = {}) => {
  2250. for (const elem of normalizeTarget(el).filter(isHTML)) {
  2251. if (isMobile && type === 'click') {
  2252. elem.addEventListener('touchstart', listener, options);
  2253. continue;
  2254. }
  2255. elem.addEventListener(type, listener, options);
  2256. }
  2257. };
  2258. /**
  2259. * @type { import("../typings/types.d.ts").formAttrs }
  2260. */
  2261. const formAttrs = (elem, attr = {}) => {
  2262. if (!elem) {
  2263. return elem;
  2264. }
  2265. for (const key in attr) {
  2266. if (typeof attr[key] === 'object') {
  2267. formAttrs(elem[key], attr[key]);
  2268. } else if (isFN(attr[key])) {
  2269. if (/^on/.test(key)) {
  2270. elem[key] = attr[key];
  2271. continue;
  2272. }
  2273. ael(elem, key, attr[key]);
  2274. } else if (key === 'class') {
  2275. elem.className = attr[key];
  2276. } else {
  2277. elem[key] = attr[key];
  2278. }
  2279. }
  2280. return elem;
  2281. };
  2282. /**
  2283. * @type { import("../typings/types.d.ts").make }
  2284. */
  2285. const make = (tagName, cname, attrs) => {
  2286. let el;
  2287. try {
  2288. const { createElement } = _self;
  2289. el = createElement(tagName);
  2290. if (!isEmpty(cname)) {
  2291. if (typeof cname === 'string') {
  2292. el.className = cname;
  2293. } else if (isObj(cname)) {
  2294. formAttrs(el, cname);
  2295. }
  2296. }
  2297. if (!isEmpty(attrs)) {
  2298. if (typeof attrs === 'string') {
  2299. el.textContent = attrs;
  2300. } else if (isObj(attrs)) {
  2301. formAttrs(el, attrs);
  2302. }
  2303. }
  2304. } catch (ex) {
  2305. if (ex instanceof DOMException) throw new Error(`${ex.name}: ${ex.message}`, { cause: 'make' });
  2306. ex.cause = 'make';
  2307. err(ex);
  2308. }
  2309. return el;
  2310. };
  2311. const $info = (() => {
  2312. if (isGM) {
  2313. if (isObj(GM.info)) {
  2314. return GM.info;
  2315. } else if (isObj(GM_info)) {
  2316. return GM_info;
  2317. }
  2318. }
  2319. return {
  2320. script: {
  2321. icon: '',
  2322. name: 'Magic Userscript+',
  2323. namespace: 'https://github.com/magicoflolis/Userscript-Plus',
  2324. updateURL: 'https://github.com/magicoflolis/Userscript-Plus/raw/master/dist/magic-userjs.js',
  2325. version: 'Bookmarklet',
  2326. bugs: 'https://github.com/magicoflolis/Userscript-Plus/issues'
  2327. }
  2328. };
  2329. })();
  2330. // #endregion
  2331. /**
  2332. * @type { import("../typings/types.d.ts").dom }
  2333. */
  2334. const dom = {
  2335. attr(target, attr, value = undefined) {
  2336. for (const elem of normalizeTarget(target).filter(isHTML)) {
  2337. if (value === undefined) {
  2338. return elem.getAttribute(attr);
  2339. }
  2340. if (value === null) {
  2341. elem.removeAttribute(attr);
  2342. } else {
  2343. elem.setAttribute(attr, value);
  2344. }
  2345. }
  2346. },
  2347. prop(target, prop, value = undefined) {
  2348. for (const elem of normalizeTarget(target).filter(isHTML)) {
  2349. if (value === undefined) {
  2350. return elem[prop];
  2351. }
  2352. elem[prop] = value;
  2353. }
  2354. },
  2355. text(target, text) {
  2356. const targets = normalizeTarget(target).filter(isHTML);
  2357. if (text === undefined) {
  2358. return targets.length !== 0 ? targets[0].textContent : undefined;
  2359. }
  2360. for (const elem of targets) {
  2361. elem.textContent = text;
  2362. }
  2363. },
  2364. remove(target) {
  2365. normalizeTarget(target)
  2366. .filter(isHTML)
  2367. .some((elem) => elem.remove());
  2368. },
  2369. cl: {
  2370. add(target, token) {
  2371. token = normalizeTarget(token, false);
  2372. return normalizeTarget(target)
  2373. .filter(isHTML)
  2374. .some((elem) => elem.classList.add(...token));
  2375. },
  2376. remove(target, token) {
  2377. token = normalizeTarget(token, false);
  2378. return normalizeTarget(target)
  2379. .filter(isHTML)
  2380. .some((elem) => elem.classList.remove(...token));
  2381. },
  2382. toggle(target, token, force) {
  2383. let r;
  2384. for (const elem of normalizeTarget(target).filter(isHTML)) {
  2385. r = elem.classList.toggle(token, force);
  2386. }
  2387. return r;
  2388. },
  2389. has(target, token) {
  2390. return normalizeTarget(target)
  2391. .filter(isHTML)
  2392. .some((elem) => elem.classList.contains(token));
  2393. }
  2394. }
  2395. };
  2396. //#region Icon SVGs
  2397. const iconSVG = {
  2398. close: {
  2399. viewBox: '0 0 384 512',
  2400. html: '<path d="M342.6 150.6c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L192 210.7 86.6 105.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3L146.7 256 41.4 361.4c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0L192 301.3 297.4 406.6c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L237.3 256 342.6 150.6z"/>'
  2401. },
  2402. code: {
  2403. viewBox: '0 0 640 512',
  2404. html: '<path d="M392.8 1.2c-17-4.9-34.7 5-39.6 22l-128 448c-4.9 17 5 34.7 22 39.6s34.7-5 39.6-22l128-448c4.9-17-5-34.7-22-39.6zm80.6 120.1c-12.5 12.5-12.5 32.8 0 45.3L562.7 256l-89.4 89.4c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0l112-112c12.5-12.5 12.5-32.8 0-45.3l-112-112c-12.5-12.5-32.8-12.5-45.3 0zm-306.7 0c-12.5-12.5-32.8-12.5-45.3 0l-112 112c-12.5 12.5-12.5 32.8 0 45.3l112 112c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L77.3 256l89.4-89.4c12.5-12.5 12.5-32.8 0-45.3z"/>'
  2405. },
  2406. collapse: {
  2407. viewBox: '0 0 448 512',
  2408. html: '<path d="M160 64c0-17.7-14.3-32-32-32s-32 14.3-32 32l0 64-64 0c-17.7 0-32 14.3-32 32s14.3 32 32 32l96 0c17.7 0 32-14.3 32-32l0-96zM32 320c-17.7 0-32 14.3-32 32s14.3 32 32 32l64 0 0 64c0 17.7 14.3 32 32 32s32-14.3 32-32l0-96c0-17.7-14.3-32-32-32l-96 0zM352 64c0-17.7-14.3-32-32-32s-32 14.3-32 32l0 96c0 17.7 14.3 32 32 32l96 0c17.7 0 32-14.3 32-32s-14.3-32-32-32l-64 0 0-64zM320 320c-17.7 0-32 14.3-32 32l0 96c0 17.7 14.3 32 32 32s32-14.3 32-32l0-64 64 0c17.7 0 32-14.3 32-32s-14.3-32-32-32l-96 0z"/>'
  2409. },
  2410. download: {
  2411. viewBox: '0 0 384 512',
  2412. html: '<path d="M64 0C28.7 0 0 28.7 0 64L0 448c0 35.3 28.7 64 64 64l256 0c35.3 0 64-28.7 64-64l0-288-128 0c-17.7 0-32-14.3-32-32L224 0 64 0zM256 0l0 128 128 0L256 0zM216 232l0 102.1 31-31c9.4-9.4 24.6-9.4 33.9 0s9.4 24.6 0 33.9l-72 72c-9.4 9.4-24.6 9.4-33.9 0l-72-72c-9.4-9.4-9.4-24.6 0-33.9s24.6-9.4 33.9 0l31 31L168 232c0-13.3 10.7-24 24-24s24 10.7 24 24z"/>'
  2413. },
  2414. expand: {
  2415. viewBox: '0 0 448 512',
  2416. html: '<path d="M32 32C14.3 32 0 46.3 0 64l0 96c0 17.7 14.3 32 32 32s32-14.3 32-32l0-64 64 0c17.7 0 32-14.3 32-32s-14.3-32-32-32L32 32zM64 352c0-17.7-14.3-32-32-32s-32 14.3-32 32l0 96c0 17.7 14.3 32 32 32l96 0c17.7 0 32-14.3 32-32s-14.3-32-32-32l-64 0 0-64zM320 32c-17.7 0-32 14.3-32 32s14.3 32 32 32l64 0 0 64c0 17.7 14.3 32 32 32s32-14.3 32-32l0-96c0-17.7-14.3-32-32-32l-96 0zM448 352c0-17.7-14.3-32-32-32s-32 14.3-32 32l0 64-64 0c-17.7 0-32 14.3-32 32s14.3 32 32 32l96 0c17.7 0 32-14.3 32-32l0-96z"/>'
  2417. },
  2418. gear: {
  2419. viewBox: '0 0 512 512',
  2420. html: '<path d="M495.9 166.6c3.2 8.7 .5 18.4-6.4 24.6l-43.3 39.4c1.1 8.3 1.7 16.8 1.7 25.4s-.6 17.1-1.7 25.4l43.3 39.4c6.9 6.2 9.6 15.9 6.4 24.6c-4.4 11.9-9.7 23.3-15.8 34.3l-4.7 8.1c-6.6 11-14 21.4-22.1 31.2c-5.9 7.2-15.7 9.6-24.5 6.8l-55.7-17.7c-13.4 10.3-28.2 18.9-44 25.4l-12.5 57.1c-2 9.1-9 16.3-18.2 17.8c-13.8 2.3-28 3.5-42.5 3.5s-28.7-1.2-42.5-3.5c-9.2-1.5-16.2-8.7-18.2-17.8l-12.5-57.1c-15.8-6.5-30.6-15.1-44-25.4L83.1 425.9c-8.8 2.8-18.6 .3-24.5-6.8c-8.1-9.8-15.5-20.2-22.1-31.2l-4.7-8.1c-6.1-11-11.4-22.4-15.8-34.3c-3.2-8.7-.5-18.4 6.4-24.6l43.3-39.4C64.6 273.1 64 264.6 64 256s.6-17.1 1.7-25.4L22.4 191.2c-6.9-6.2-9.6-15.9-6.4-24.6c4.4-11.9 9.7-23.3 15.8-34.3l4.7-8.1c6.6-11 14-21.4 22.1-31.2c5.9-7.2 15.7-9.6 24.5-6.8l55.7 17.7c13.4-10.3 28.2-18.9 44-25.4l12.5-57.1c2-9.1 9-16.3 18.2-17.8C227.3 1.2 241.5 0 256 0s28.7 1.2 42.5 3.5c9.2 1.5 16.2 8.7 18.2 17.8l12.5 57.1c15.8 6.5 30.6 15.1 44 25.4l55.7-17.7c8.8-2.8 18.6-.3 24.5 6.8c8.1 9.8 15.5 20.2 22.1 31.2l4.7 8.1c6.1 11 11.4 22.4 15.8 34.3zM256 336a80 80 0 1 0 0-160 80 80 0 1 0 0 160z"/>'
  2421. },
  2422. github: {
  2423. viewBox: '0 0 496 512',
  2424. html: '<path d="M165.9 397.4c0 2-2.3 3.6-5.2 3.6-3.3 .3-5.6-1.3-5.6-3.6 0-2 2.3-3.6 5.2-3.6 3-.3 5.6 1.3 5.6 3.6zm-31.1-4.5c-.7 2 1.3 4.3 4.3 4.9 2.6 1 5.6 0 6.2-2s-1.3-4.3-4.3-5.2c-2.6-.7-5.5 .3-6.2 2.3zm44.2-1.7c-2.9 .7-4.9 2.6-4.6 4.9 .3 2 2.9 3.3 5.9 2.6 2.9-.7 4.9-2.6 4.6-4.6-.3-1.9-3-3.2-5.9-2.9zM244.8 8C106.1 8 0 113.3 0 252c0 110.9 69.8 205.8 169.5 239.2 12.8 2.3 17.3-5.6 17.3-12.1 0-6.2-.3-40.4-.3-61.4 0 0-70 15-84.7-29.8 0 0-11.4-29.1-27.8-36.6 0 0-22.9-15.7 1.6-15.4 0 0 24.9 2 38.6 25.8 21.9 38.6 58.6 27.5 72.9 20.9 2.3-16 8.8-27.1 16-33.7-55.9-6.2-112.3-14.3-112.3-110.5 0-27.5 7.6-41.3 23.6-58.9-2.6-6.5-11.1-33.3 2.6-67.9 20.9-6.5 69 27 69 27 20-5.6 41.5-8.5 62.8-8.5s42.8 2.9 62.8 8.5c0 0 48.1-33.6 69-27 13.7 34.7 5.2 61.4 2.6 67.9 16 17.7 25.8 31.5 25.8 58.9 0 96.5-58.9 104.2-114.8 110.5 9.2 7.9 17 22.9 17 46.4 0 33.7-.3 75.4-.3 83.6 0 6.5 4.6 14.4 17.3 12.1C428.2 457.8 496 362.9 496 252 496 113.3 383.5 8 244.8 8zM97.2 352.9c-1.3 1-1 3.3 .7 5.2 1.6 1.6 3.9 2.3 5.2 1 1.3-1 1-3.3-.7-5.2-1.6-1.6-3.9-2.3-5.2-1zm-10.8-8.1c-.7 1.3 .3 2.9 2.3 3.9 1.6 1 3.6 .7 4.3-.7 .7-1.3-.3-2.9-2.3-3.9-2-.6-3.6-.3-4.3 .7zm32.4 35.6c-1.6 1.3-1 4.3 1.3 6.2 2.3 2.3 5.2 2.6 6.5 1 1.3-1.3 .7-4.3-1.3-6.2-2.2-2.3-5.2-2.6-6.5-1zm-11.4-14.7c-1.6 1-1.6 3.6 0 5.9 1.6 2.3 4.3 3.3 5.6 2.3 1.6-1.3 1.6-3.9 0-6.2-1.4-2.3-4-3.3-5.6-2z"/>'
  2425. },
  2426. globe: {
  2427. viewBox: '0 0 512 512',
  2428. html: '<path d="M352 256c0 22.2-1.2 43.6-3.3 64l-185.3 0c-2.2-20.4-3.3-41.8-3.3-64s1.2-43.6 3.3-64l185.3 0c2.2 20.4 3.3 41.8 3.3 64zm28.8-64l123.1 0c5.3 20.5 8.1 41.9 8.1 64s-2.8 43.5-8.1 64l-123.1 0c2.1-20.6 3.2-42 3.2-64s-1.1-43.4-3.2-64zm112.6-32l-116.7 0c-10-63.9-29.8-117.4-55.3-151.6c78.3 20.7 142 77.5 171.9 151.6zm-149.1 0l-176.6 0c6.1-36.4 15.5-68.6 27-94.7c10.5-23.6 22.2-40.7 33.5-51.5C239.4 3.2 248.7 0 256 0s16.6 3.2 27.8 13.8c11.3 10.8 23 27.9 33.5 51.5c11.6 26 20.9 58.2 27 94.7zm-209 0L18.6 160C48.6 85.9 112.2 29.1 190.6 8.4C165.1 42.6 145.3 96.1 135.3 160zM8.1 192l123.1 0c-2.1 20.6-3.2 42-3.2 64s1.1 43.4 3.2 64L8.1 320C2.8 299.5 0 278.1 0 256s2.8-43.5 8.1-64zM194.7 446.6c-11.6-26-20.9-58.2-27-94.6l176.6 0c-6.1 36.4-15.5 68.6-27 94.6c-10.5 23.6-22.2 40.7-33.5 51.5C272.6 508.8 263.3 512 256 512s-16.6-3.2-27.8-13.8c-11.3-10.8-23-27.9-33.5-51.5zM135.3 352c10 63.9 29.8 117.4 55.3 151.6C112.2 482.9 48.6 426.1 18.6 352l116.7 0zm358.1 0c-30 74.1-93.6 130.9-171.9 151.6c25.5-34.2 45.2-87.7 55.3-151.6l116.7 0z"/>'
  2429. },
  2430. install: {
  2431. viewBox: '0 0 512 512',
  2432. html: '<path d="M288 32c0-17.7-14.3-32-32-32s-32 14.3-32 32l0 242.7-73.4-73.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l128 128c12.5 12.5 32.8 12.5 45.3 0l128-128c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L288 274.7 288 32zM64 352c-35.3 0-64 28.7-64 64l0 32c0 35.3 28.7 64 64 64l384 0c35.3 0 64-28.7 64-64l0-32c0-35.3-28.7-64-64-64l-101.5 0-45.3 45.3c-25 25-65.5 25-90.5 0L165.5 352 64 352zm368 56a24 24 0 1 1 0 48 24 24 0 1 1 0-48z"/>'
  2433. },
  2434. issue: {
  2435. viewBox: '0 0 512 512',
  2436. html: '<path d="M256 0c53 0 96 43 96 96l0 3.6c0 15.7-12.7 28.4-28.4 28.4l-135.1 0c-15.7 0-28.4-12.7-28.4-28.4l0-3.6c0-53 43-96 96-96zM41.4 105.4c12.5-12.5 32.8-12.5 45.3 0l64 64c.7 .7 1.3 1.4 1.9 2.1c14.2-7.3 30.4-11.4 47.5-11.4l112 0c17.1 0 33.2 4.1 47.5 11.4c.6-.7 1.2-1.4 1.9-2.1l64-64c12.5-12.5 32.8-12.5 45.3 0s12.5 32.8 0 45.3l-64 64c-.7 .7-1.4 1.3-2.1 1.9c6.2 12 10.1 25.3 11.1 39.5l64.3 0c17.7 0 32 14.3 32 32s-14.3 32-32 32l-64 0c0 24.6-5.5 47.8-15.4 68.6c2.2 1.3 4.2 2.9 6 4.8l64 64c12.5 12.5 12.5 32.8 0 45.3s-32.8 12.5-45.3 0l-63.1-63.1c-24.5 21.8-55.8 36.2-90.3 39.6L272 240c0-8.8-7.2-16-16-16s-16 7.2-16 16l0 239.2c-34.5-3.4-65.8-17.8-90.3-39.6L86.6 502.6c-12.5 12.5-32.8 12.5-45.3 0s-12.5-32.8 0-45.3l64-64c1.9-1.9 3.9-3.4 6-4.8C101.5 367.8 96 344.6 96 320l-64 0c-17.7 0-32-14.3-32-32s14.3-32 32-32l64.3 0c1.1-14.1 5-27.5 11.1-39.5c-.7-.6-1.4-1.2-2.1-1.9l-64-64c-12.5-12.5-12.5-32.8 0-45.3z"/>'
  2437. },
  2438. minus: {
  2439. viewBox: '0 0 448 512',
  2440. html: '<path d="M432 256c0 17.7-14.3 32-32 32L48 288c-17.7 0-32-14.3-32-32s14.3-32 32-32l352 0c17.7 0 32 14.3 32 32z"/>'
  2441. },
  2442. nav: {
  2443. viewBox: '0 0 448 512',
  2444. html: '<path d="M0 96C0 78.3 14.3 64 32 64l384 0c17.7 0 32 14.3 32 32s-14.3 32-32 32L32 128C14.3 128 0 113.7 0 96zM0 256c0-17.7 14.3-32 32-32l384 0c17.7 0 32 14.3 32 32s-14.3 32-32 32L32 288c-17.7 0-32-14.3-32-32zM448 416c0 17.7-14.3 32-32 32L32 448c-17.7 0-32-14.3-32-32s14.3-32 32-32l384 0c17.7 0 32 14.3 32 32z"/>'
  2445. },
  2446. pager: {
  2447. viewBox: '0 0 512 512',
  2448. html: '<path d="M0 128C0 92.7 28.7 64 64 64l384 0c35.3 0 64 28.7 64 64l0 256c0 35.3-28.7 64-64 64L64 448c-35.3 0-64-28.7-64-64L0 128zm64 32l0 64c0 17.7 14.3 32 32 32l320 0c17.7 0 32-14.3 32-32l0-64c0-17.7-14.3-32-32-32L96 128c-17.7 0-32 14.3-32 32zM80 320c-13.3 0-24 10.7-24 24s10.7 24 24 24l56 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-56 0zm136 0c-13.3 0-24 10.7-24 24s10.7 24 24 24l48 0c13.3 0 24-10.7 24-24s-10.7-24-24-24l-48 0z"/>'
  2449. },
  2450. verified: {
  2451. viewBox: '0 0 56 56',
  2452. fill: 'currentColor',
  2453. stroke: 'currentColor',
  2454. html: '<g stroke-width="0"/><g stroke-linecap="round" stroke-linejoin="round"/><g><path d="M 23.6641 52.3985 C 26.6407 55.375 29.3594 55.3516 32.3126 52.3985 L 35.9219 48.8125 C 36.2969 48.4610 36.6250 48.3203 37.1172 48.3203 L 42.1797 48.3203 C 46.3749 48.3203 48.3204 46.3985 48.3204 42.1797 L 48.3204 37.1172 C 48.3204 36.625 48.4610 36.2969 48.8124 35.9219 L 52.3749 32.3125 C 55.3749 29.3594 55.3514 26.6407 52.3749 23.6641 L 48.8124 20.0547 C 48.4610 19.7031 48.3204 19.3516 48.3204 18.8829 L 48.3204 13.7969 C 48.3204 9.625 46.3985 7.6563 42.1797 7.6563 L 37.1172 7.6563 C 36.6250 7.6563 36.2969 7.5391 35.9219 7.1875 L 32.3126 3.6016 C 29.3594 .6250 26.6407 .6485 23.6641 3.6016 L 20.0547 7.1875 C 19.7032 7.5391 19.3516 7.6563 18.8828 7.6563 L 13.7969 7.6563 C 9.6016 7.6563 7.6563 9.5782 7.6563 13.7969 L 7.6563 18.8829 C 7.6563 19.3516 7.5391 19.7031 7.1876 20.0547 L 3.6016 23.6641 C .6251 26.6407 .6485 29.3594 3.6016 32.3125 L 7.1876 35.9219 C 7.5391 36.2969 7.6563 36.625 7.6563 37.1172 L 7.6563 42.1797 C 7.6563 46.3750 9.6016 48.3203 13.7969 48.3203 L 18.8828 48.3203 C 19.3516 48.3203 19.7032 48.4610 20.0547 48.8125 Z M 26.2891 49.7734 L 21.8828 45.3438 C 21.3672 44.8047 20.8282 44.5938 20.1016 44.5938 L 13.7969 44.5938 C 11.7110 44.5938 11.3828 44.2656 11.3828 42.1797 L 11.3828 35.875 C 11.3828 35.1719 11.1719 34.6329 10.6563 34.1172 L 6.2266 29.7109 C 4.7501 28.2109 4.7501 27.7891 6.2266 26.2891 L 10.6563 21.8829 C 11.1719 21.3672 11.3828 20.8282 11.3828 20.1016 L 11.3828 13.7969 C 11.3828 11.6875 11.6876 11.3829 13.7969 11.3829 L 20.1016 11.3829 C 20.8282 11.3829 21.3672 11.1953 21.8828 10.6563 L 26.2891 6.2266 C 27.7891 4.7500 28.2110 4.7500 29.7110 6.2266 L 34.1172 10.6563 C 34.6328 11.1953 35.1719 11.3829 35.8750 11.3829 L 42.1797 11.3829 C 44.2657 11.3829 44.5938 11.7109 44.5938 13.7969 L 44.5938 20.1016 C 44.5938 20.8282 44.8282 21.3672 45.3439 21.8829 L 49.7733 26.2891 C 51.2498 27.7891 51.2498 28.2109 49.7733 29.7109 L 45.3439 34.1172 C 44.8282 34.6329 44.5938 35.1719 44.5938 35.875 L 44.5938 42.1797 C 44.5938 44.2656 44.2657 44.5938 42.1797 44.5938 L 35.8750 44.5938 C 35.1719 44.5938 34.6328 44.8047 34.1172 45.3438 L 29.7110 49.7734 C 28.2110 51.2500 27.7891 51.2500 26.2891 49.7734 Z M 24.3438 39.2266 C 25.0235 39.2266 25.5391 38.9453 25.8907 38.5234 L 38.8985 20.3360 C 39.1563 19.9609 39.2969 19.5391 39.2969 19.1407 C 39.2969 18.1094 38.5001 17.2891 37.4219 17.2891 C 36.6485 17.2891 36.2266 17.5469 35.7579 18.2266 L 24.2735 34.3985 L 18.3438 27.8594 C 17.9454 27.4141 17.5001 27.2266 16.9141 27.2266 C 15.7657 27.2266 14.9454 28.0000 14.9454 29.0782 C 14.9454 29.5469 15.1094 29.9922 15.4376 30.3203 L 22.8907 38.6172 C 23.2423 38.9922 23.6876 39.2266 24.3438 39.2266 Z"/></g>'
  2455. },
  2456. refresh: {
  2457. viewBox: '0 0 512 512',
  2458. fill: 'currentColor',
  2459. html: '<path d="M463.5 224l8.5 0c13.3 0 24-10.7 24-24l0-128c0-9.7-5.8-18.5-14.8-22.2s-19.3-1.7-26.2 5.2L413.4 96.6c-87.6-86.5-228.7-86.2-315.8 1c-87.5 87.5-87.5 229.3 0 316.8s229.3 87.5 316.8 0c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0c-62.5 62.5-163.8 62.5-226.3 0s-62.5-163.8 0-226.3c62.2-62.2 162.7-62.5 225.3-1L327 183c-6.9 6.9-8.9 17.2-5.2 26.2s12.5 14.8 22.2 14.8l119.5 0z"/>'
  2460. },
  2461. load(type, container) {
  2462. const { createElementNS } = _self;
  2463. const svgElem = createElementNS('http://www.w3.org/2000/svg', 'svg');
  2464. for (const [k, v] of Object.entries(iconSVG[type])) {
  2465. if (k === 'html') {
  2466. continue;
  2467. }
  2468. svgElem.setAttributeNS(null, k, v);
  2469. }
  2470. try {
  2471. if (typeof iconSVG[type].html === 'string') {
  2472. svgElem.innerHTML = iconSVG[type].html;
  2473. svgElem.setAttribute('id', `mujs_${type ?? 'Unknown'}`);
  2474. }
  2475. // eslint-disable-next-line no-unused-vars
  2476. } catch (ex) {
  2477. /* empty */
  2478. }
  2479. if (container) {
  2480. container.appendChild(svgElem);
  2481. return svgElem;
  2482. }
  2483. return svgElem.outerHTML;
  2484. }
  2485. };
  2486. //#endregion
  2487. /**
  2488. * @type { import("../typings/UserJS.d.ts").StorageSystem }
  2489. */
  2490. const StorageSystem = {
  2491. prefix: 'MUJS',
  2492. events: new Set(),
  2493. getItem(key) {
  2494. return window.localStorage.getItem(key);
  2495. },
  2496. has(key) {
  2497. return !this.getItem(key);
  2498. },
  2499. setItem(key, value) {
  2500. window.localStorage.setItem(key, value);
  2501. },
  2502. remove(key) {
  2503. window.localStorage.removeItem(key);
  2504. },
  2505. async setValue(key, v) {
  2506. if (!v) {
  2507. return;
  2508. }
  2509. v = typeof v === 'string' ? v : JSON.stringify(v);
  2510. if (isGM) {
  2511. if (isFN(GM.setValue)) {
  2512. await GM.setValue(key, v);
  2513. } else if (isFN(GM_setValue)) {
  2514. GM_setValue(key, v);
  2515. }
  2516. } else {
  2517. this.setItem(`${this.prefix}-${key}`, v);
  2518. }
  2519. },
  2520. async getValue(key, def = {}) {
  2521. try {
  2522. if (isGM) {
  2523. let GMType;
  2524. if (isFN(GM.getValue)) {
  2525. GMType = await GM.getValue(key, JSON.stringify(def));
  2526. } else if (isFN(GM_getValue)) {
  2527. GMType = GM_getValue(key, JSON.stringify(def));
  2528. }
  2529. if (!isNull(GMType)) {
  2530. return JSON.parse(GMType);
  2531. }
  2532. }
  2533. return this.has(`${this.prefix}-${key}`)
  2534. ? JSON.parse(this.getItem(`${this.prefix}-${key}`))
  2535. : def;
  2536. } catch (ex) {
  2537. ex.cause = 'getValue';
  2538. err(ex);
  2539. return def;
  2540. }
  2541. }
  2542. };
  2543. const Command = {
  2544. cmds: new Set(),
  2545. register(text, command) {
  2546. if (!isGM) {
  2547. return;
  2548. }
  2549. if (isFN(command)) {
  2550. if (this.cmds.has(command)) {
  2551. return;
  2552. }
  2553. this.cmds.add(command);
  2554. }
  2555. if (isFN(GM.registerMenuCommand)) {
  2556. GM.registerMenuCommand(text, command);
  2557. } else if (isFN(GM_registerMenuCommand)) {
  2558. GM_registerMenuCommand(text, command);
  2559. }
  2560. }
  2561. };
  2562. /**
  2563. * @type { import("../typings/UserJS.d.ts").Network }
  2564. */
  2565. const Network = {
  2566. async req(url, method = 'GET', responseType = 'json', data, useFetch = false) {
  2567. if (isEmpty(url)) {
  2568. throw new Error('"url" parameter is empty');
  2569. }
  2570. data = Object.assign({}, data);
  2571. method = this.bscStr(method, false);
  2572. responseType = this.bscStr(responseType, true);
  2573. const params = {
  2574. method,
  2575. ...data
  2576. };
  2577. if (isGM && !useFetch) {
  2578. if (params.credentials) {
  2579. Object.assign(params, {
  2580. anonymous: false
  2581. });
  2582. if (Object.is(params.credentials, 'omit')) {
  2583. Object.assign(params, {
  2584. anonymous: true
  2585. });
  2586. }
  2587. delete params.credentials;
  2588. }
  2589. } else if (params.onprogress) {
  2590. delete params.onprogress;
  2591. }
  2592. return new Promise((resolve, reject) => {
  2593. if (isGM && !useFetch) {
  2594. Network.xmlRequest({
  2595. url,
  2596. responseType,
  2597. ...params,
  2598. onerror: (r_1) => {
  2599. reject(new Error(`${r_1.status} ${url}`));
  2600. },
  2601. onload: (r_1) => {
  2602. if (r_1.status !== 200) reject(new Error(`${r_1.status} ${url}`));
  2603. if (responseType.match(/basic/)) resolve(r_1);
  2604. resolve(r_1.response);
  2605. }
  2606. });
  2607. } else {
  2608. fetch(url, params)
  2609. .then((response_1) => {
  2610. if (!response_1.ok) reject(response_1);
  2611. const check = (str_2 = 'text') => {
  2612. return isFN(response_1[str_2]) ? response_1[str_2]() : response_1;
  2613. };
  2614. if (responseType.match(/buffer/)) {
  2615. resolve(check('arrayBuffer'));
  2616. } else if (responseType.match(/json/)) {
  2617. resolve(check('json'));
  2618. } else if (responseType.match(/text/)) {
  2619. resolve(check('text'));
  2620. } else if (responseType.match(/blob/)) {
  2621. resolve(check('blob'));
  2622. } else if (responseType.match(/formdata/)) {
  2623. resolve(check('formData'));
  2624. } else if (responseType.match(/clone/)) {
  2625. resolve(check('clone'));
  2626. } else if (responseType.match(/document/)) {
  2627. const respTxt = check('text');
  2628. const domParser = new DOMParser();
  2629. if (respTxt instanceof Promise) {
  2630. respTxt.then((txt) => {
  2631. const doc = domParser.parseFromString(txt, 'text/html');
  2632. resolve(doc);
  2633. });
  2634. } else {
  2635. const doc = domParser.parseFromString(respTxt, 'text/html');
  2636. resolve(doc);
  2637. }
  2638. } else {
  2639. resolve(response_1);
  2640. }
  2641. })
  2642. .catch(reject);
  2643. }
  2644. });
  2645. },
  2646. format(bytes, decimals = 2) {
  2647. if (Number.isNaN(bytes)) return `0 ${this.sizes[0]}`;
  2648. const k = ####;
  2649. const dm = decimals < 0 ? 0 : decimals;
  2650. const i = Math.floor(Math.log(bytes) / Math.log(k));
  2651. return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${this.sizes[i]}`;
  2652. },
  2653. sizes: ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'],
  2654. async xmlRequest(details) {
  2655. if (isGM) {
  2656. if (isFN(GM.xmlHttpRequest)) {
  2657. return GM.xmlHttpRequest(details);
  2658. } else if (isFN(GM_xmlhttpRequest)) {
  2659. return GM_xmlhttpRequest(details);
  2660. }
  2661. }
  2662. return await new Promise((resolve, reject) => {
  2663. const { XMLHttpRequest } = _self;
  2664. const req = new XMLHttpRequest();
  2665. let method = 'GET';
  2666. let url = BLANK_PAGE;
  2667. let body;
  2668. for (const [key, value] of Object.entries(details)) {
  2669. if (key === 'onload') {
  2670. req.addEventListener('load', () => {
  2671. if (isFN(value)) {
  2672. value(req);
  2673. }
  2674. resolve(req);
  2675. });
  2676. } else if (key === 'onerror') {
  2677. req.addEventListener('error', (evt) => {
  2678. if (isFN(value)) {
  2679. value(evt);
  2680. }
  2681. reject(evt);
  2682. });
  2683. } else if (key === 'onabort') {
  2684. req.addEventListener('abort', (evt) => {
  2685. if (isFN(value)) {
  2686. value(evt);
  2687. }
  2688. reject(evt);
  2689. });
  2690. } else if (key === 'onprogress') {
  2691. req.addEventListener('progress', value);
  2692. } else if (key === 'responseType') {
  2693. if (value === 'buffer') {
  2694. req.responseType = 'arraybuffer';
  2695. } else {
  2696. req.responseType = value;
  2697. }
  2698. } else if (key === 'method') {
  2699. method = value;
  2700. } else if (key === 'url') {
  2701. url = value;
  2702. } else if (key === 'body') {
  2703. body = value;
  2704. }
  2705. }
  2706. req.open(method, url);
  2707. if (isEmpty(req.responseType)) {
  2708. req.responseType = 'text';
  2709. }
  2710. if (body) {
  2711. req.send(body);
  2712. } else {
  2713. req.send();
  2714. }
  2715. });
  2716. },
  2717. bscStr(str = '', lowerCase = true) {
  2718. const txt = str[lowerCase ? 'toLowerCase' : 'toUpperCase']();
  2719. return txt.replaceAll(/\W/g, '');
  2720. }
  2721. };
  2722. const Counter = {
  2723. cnt: {
  2724. total: {
  2725. count: 0
  2726. }
  2727. },
  2728. set(engine) {
  2729. if (!this.cnt[engine.name]) {
  2730. const counter = make('count-frame', engine.enabled ? '' : 'hidden', {
  2731. dataset: {
  2732. counter: engine.name
  2733. },
  2734. title: decode(engine.query ?? engine.url),
  2735. textContent: '0'
  2736. });
  2737. this.cnt[engine.name] = {
  2738. root: counter,
  2739. count: 0
  2740. };
  2741. return counter;
  2742. }
  2743. return this.cnt[engine.name].root;
  2744. },
  2745. update(count, engine) {
  2746. this.cnt[engine.name].count += count;
  2747. this.cnt.total.count += count;
  2748. this.updateAll();
  2749. },
  2750. updateAll() {
  2751. for (const v of Object.values(this.cnt)) dom.text(v.root, v.count);
  2752. },
  2753. reset() {
  2754. for (const [k, v] of Object.entries(this.cnt)) {
  2755. dom.text(v.root, 0);
  2756. v.count = 0;
  2757. const engine = cfg.engines.find((engine) => k === engine.name);
  2758. if (engine) {
  2759. dom.cl[engine.enabled ? 'remove' : 'add'](v.root, 'hidden');
  2760. }
  2761. }
  2762. }
  2763. };
  2764. // #region Container
  2765. /**
  2766. * @type { import("../typings/UserJS.d.ts").Container }
  2767. */
  2768. class Container {
  2769. webpage;
  2770. host;
  2771. ready;
  2772. injected;
  2773. shadowRoot;
  2774. supported;
  2775. frame;
  2776. userjsCache;
  2777. root;
  2778. unsaved;
  2779. isBlacklisted;
  2780. rebuild;
  2781. opacityMin;
  2782. opacityMax;
  2783. constructor() {
  2784. this.remove = this.remove.bind(this);
  2785. this.refresh = this.refresh.bind(this);
  2786. this.showError = this.showError.bind(this);
  2787. this.toElem = this.toElem.bind(this);
  2788. this.webpage = url;
  2789. this.host = getHostname(url?.hostname ?? BLANK_PAGE);
  2790. this.ready = false;
  2791. this.injected = false;
  2792. this.shadowRoot = undefined;
  2793. this.supported = isFN(make('main-userjs').attachShadow);
  2794. this.frame = this.supported
  2795. ? make('main-userjs', {
  2796. dataset: {
  2797. insertedBy: $info.script.name,
  2798. role: 'primary-container'
  2799. }
  2800. })
  2801. : make('iframe', 'mujs-iframe', {
  2802. dataset: {
  2803. insertedBy: $info.script.name,
  2804. role: 'primary-iframe'
  2805. },
  2806. loading: 'lazy',
  2807. src: BLANK_PAGE,
  2808. style:
  2809. 'position: fixed;bottom: 1rem;right: 1rem;height: 525px;width: 90%;margin: 0px 1rem;z-index: 100000000000000020 !important;',
  2810. onload: (iFrame) => {
  2811. /**
  2812. * @type { HTMLIFrameElement }
  2813. */
  2814. const target = iFrame.target;
  2815. if (!target.contentDocument) {
  2816. return;
  2817. }
  2818. this.shadowRoot = target.contentDocument.documentElement;
  2819. this.ready = true;
  2820. dom.cl.add([this.shadowRoot, target.contentDocument.body], 'mujs-iframe');
  2821. }
  2822. });
  2823. if (this.supported) {
  2824. this.shadowRoot = this.frame.attachShadow({ mode: 'closed' });
  2825. this.ready = true;
  2826. }
  2827. this.hostCache = new Map();
  2828. this.userjsCache = new Map();
  2829. this.root = make('mujs-root');
  2830. this.unsaved = false;
  2831. this.isBlacklisted = false;
  2832. this.rebuild = false;
  2833. this.opacityMin = '0.15';
  2834. this.opacityMax = '1';
  2835. this.elementsReady = this.init();
  2836. const Timeout = class {
  2837. constructor() {
  2838. this.ids = [];
  2839. }
  2840. set(delay, reason) {
  2841. const { setTimeout } = _self;
  2842. return new Promise((resolve, reject) => {
  2843. const id = setTimeout(() => {
  2844. Object.is(reason, null) || Object.is(reason, undefined) ? resolve() : reject(reason);
  2845. this.clear(id);
  2846. }, delay);
  2847. this.ids.push(id);
  2848. });
  2849. }
  2850. clear(...ids) {
  2851. const { clearTimeout } = _self;
  2852. this.ids = this.ids.filter((id) => {
  2853. if (ids.includes(id)) {
  2854. clearTimeout(id);
  2855. return false;
  2856. }
  2857. return true;
  2858. });
  2859. }
  2860. };
  2861. this.timeouts = {
  2862. frame: new Timeout(),
  2863. mouse: new Timeout()
  2864. };
  2865. this.injFN = BLANK_FN;
  2866. window.addEventListener('beforeunload', this.remove);
  2867. }
  2868. /**
  2869. * @param { function(): * } callback
  2870. * @param { Document } doc
  2871. */
  2872. async inject(callback, doc) {
  2873. if (this.checkBlacklist(this.host)) {
  2874. err(`Blacklisted "${this.host}"`);
  2875. this.remove();
  2876. return;
  2877. }
  2878. if (!this.shadowRoot) {
  2879. return;
  2880. }
  2881. if (doc === null) {
  2882. return;
  2883. }
  2884. while (this.ready === false) {
  2885. await new Promise((resolve) => requestAnimationFrame(resolve));
  2886. }
  2887. try {
  2888. doc.documentElement.appendChild(this.frame);
  2889. if (this.injected) {
  2890. if (isFN(this.injFN.build)) {
  2891. this.injFN.build();
  2892. }
  2893. return;
  2894. }
  2895. this.shadowRoot.append(this.root);
  2896. if (isNull(this.loadCSS(main_css, 'primary-stylesheet')))
  2897. throw new Error('Failed to initialize script!', { cause: 'loadCSS' });
  2898. this.injected = true;
  2899. this.initFn();
  2900. if (isFN(callback) && this.elementsReady) this.injFN = callback.call(this, this.shadowRoot);
  2901. } catch (ex) {
  2902. err(ex);
  2903. this.remove();
  2904. }
  2905. }
  2906. initFn() {
  2907. this.setTheme();
  2908. Counter.cnt.total.root = this.mainbtn;
  2909. if (this.countframe)
  2910. for (const engine of cfg.engines) this.countframe.append(Counter.set(engine));
  2911. const { cfgpage, table, supported, frame, refresh, hostCache, urlBar, host } = this;
  2912. class Tabs {
  2913. /**
  2914. * @param { HTMLElement } root
  2915. */
  2916. constructor(root) {
  2917. /**
  2918. * @type { Set<HTMLElement> }
  2919. */
  2920. this.pool = new Set();
  2921. this.blank = BLANK_PAGE;
  2922. this.protocal = 'mujs:';
  2923. this.protoReg = new RegExp(`${this.protocal}(.+)`, 'i');
  2924. this.el = {
  2925. add: make('mujs-addtab', {
  2926. textContent: '+',
  2927. dataset: {
  2928. command: 'new-tab'
  2929. }
  2930. }),
  2931. head: make('mujs-tabs'),
  2932. root
  2933. };
  2934. this.el.head.append(this.el.add);
  2935. this.el.root.append(this.el.head);
  2936. this.custom = BLANK_FN;
  2937. }
  2938. /**
  2939. * @param {string} hostname
  2940. */
  2941. getTab(hostname) {
  2942. return [...this.pool].find(({ dataset }) => hostname === dataset.host);
  2943. }
  2944. getActive() {
  2945. return [...this.pool].find((tab) => tab.classList.contains('active'));
  2946. }
  2947. /**
  2948. * @param {string} hostname
  2949. */
  2950. intFN(hostname) {
  2951. const p = this.protoReg.exec(hostname);
  2952. if (!p) {
  2953. return;
  2954. }
  2955. if (p[1] === 'settings') {
  2956. dom.cl.remove(cfgpage, 'hidden');
  2957. dom.cl.add(table, 'hidden');
  2958. urlBar.placeholder = 'Search settings';
  2959. if (!supported) {
  2960. dom.attr(frame, 'style', 'height: 100%;');
  2961. }
  2962. }
  2963. }
  2964. /**
  2965. * @param {HTMLElement} tab
  2966. * @param {boolean} [build]
  2967. */
  2968. active(tab, build = true) {
  2969. if (!this.pool.has(tab)) this.pool.add(tab);
  2970. dom.cl.add([table, cfgpage], 'hidden');
  2971. dom.cl.remove([...this.pool], 'active');
  2972. dom.cl.add(tab, 'active');
  2973. if (!build) {
  2974. dom.cl.remove(table, 'hidden');
  2975. return;
  2976. }
  2977. const host = tab.dataset.host ?? this.blank;
  2978. if (host === this.blank) {
  2979. dom.cl.add(cfgpage, 'hidden');
  2980. dom.cl.remove(table, 'hidden');
  2981. refresh();
  2982. } else if (host.startsWith(this.protocal)) {
  2983. this.intFN(host);
  2984. } else {
  2985. dom.cl.add(cfgpage, 'hidden');
  2986. dom.cl.remove(table, 'hidden');
  2987. this.custom(host);
  2988. }
  2989. }
  2990. /** @param { HTMLElement } tab */
  2991. close(tab) {
  2992. if (this.pool.has(tab)) this.pool.delete(tab);
  2993. const host = tab.dataset.host;
  2994. if (cfg.clearTabCache && hostCache.has(host)) hostCache.delete(host);
  2995. if (tab.classList.contains('active')) refresh();
  2996. const sibling = tab.nextElementSibling ?? tab.previousElementSibling;
  2997. if (sibling) {
  2998. if (sibling.dataset.command !== 'new-tab') {
  2999. this.active(sibling);
  3000. }
  3001. }
  3002. tab.remove();
  3003. }
  3004. /**
  3005. * @param {string} [hostname]
  3006. */
  3007. create(hostname = undefined) {
  3008. if (typeof hostname === 'string') {
  3009. const createdTab = this.getTab(hostname);
  3010. if (this.protoReg.test(hostname) && createdTab) {
  3011. this.active(createdTab);
  3012. return;
  3013. }
  3014. }
  3015. const tab = make('mujs-tab', {
  3016. dataset: {
  3017. command: 'switch-tab'
  3018. },
  3019. style: `order: ${this.el.head.childElementCount};`
  3020. });
  3021. const tabClose = make('mu-js', {
  3022. dataset: {
  3023. command: 'close-tab'
  3024. },
  3025. title: i18n$('close'),
  3026. textContent: 'X'
  3027. });
  3028. const tabHost = make('mujs-host');
  3029. const p = this.protoReg.exec(hostname);
  3030. tab.append(tabHost, tabClose);
  3031. this.el.head.append(tab);
  3032. this.active(tab, false);
  3033. if (isNull(hostname)) {
  3034. refresh();
  3035. tab.dataset.host = this.blank;
  3036. tabHost.title = i18n$('newTab');
  3037. tabHost.textContent = i18n$('newTab');
  3038. } else if (p) {
  3039. tab.dataset.host = hostname || host;
  3040. tabHost.title = p[1] || tab.dataset.host;
  3041. tabHost.textContent = tabHost.title;
  3042. this.intFN(hostname);
  3043. } else {
  3044. tab.dataset.host = hostname || host;
  3045. tabHost.title = hostname || host;
  3046. tabHost.textContent = tabHost.title;
  3047. }
  3048. return tab;
  3049. }
  3050. }
  3051. this.Tabs = new Tabs(this.toolbar);
  3052. this.Tabs.create(host);
  3053. const tabbody = this.tabbody;
  3054. const getCellValue = (tr, idx) =>
  3055. tr.children[idx].dataset.value || tr.children[idx].textContent;
  3056. const comparer = (idx, asc) => (a, b) =>
  3057. ((v1, v2) =>
  3058. v1 !== '' && v2 !== '' && !isNaN(v1) && !isNaN(v2)
  3059. ? v1 - v2
  3060. : v1.toString().localeCompare(v2))(
  3061. getCellValue(asc ? a : b, idx),
  3062. getCellValue(asc ? b : a, idx)
  3063. );
  3064. for (const th of this.tabhead.rows[0].cells) {
  3065. if (dom.text(th) === i18n$('install')) continue;
  3066. dom.cl.add(th, 'mujs-pointer');
  3067. ael(th, 'click', () => {
  3068. /** [Stack Overflow Reference](https://stackoverflow.com/questions/14267781/sorting-html-table-with-javascript/53880407#53880407) */
  3069. Array.from(tabbody.querySelectorAll('tr'))
  3070. .sort(comparer(Array.from(th.parentNode.children).indexOf(th), (this.asc = !this.asc)))
  3071. .forEach((tr) => tabbody.appendChild(tr));
  3072. });
  3073. }
  3074. }
  3075. init() {
  3076. try {
  3077. // #region Elements
  3078. this.mainframe = make('mu-js', 'mainframe', {
  3079. style: `opacity: ${this.opacityMin};`
  3080. });
  3081. this.countframe = make('mujs-column');
  3082. this.mainbtn = make('count-frame', 'mainbtn', {
  3083. textContent: '0'
  3084. });
  3085. this.urlBar = make('input', 'mujs-url-bar', {
  3086. autocomplete: 'off',
  3087. spellcheck: false,
  3088. type: 'text',
  3089. placeholder: i18n$('search_placeholder')
  3090. });
  3091. this.rateContainer = make('mujs-column', 'rate-container');
  3092. this.footer = make('mujs-row', 'mujs-footer');
  3093. this.tabbody = make('tbody');
  3094. this.promptElem = make('mujs-row', 'mujs-prompt');
  3095. this.toolbar = make('mujs-toolbar');
  3096. this.table = make('table');
  3097. this.tabhead = make('thead');
  3098. this.header = make('mujs-header');
  3099. this.tbody = make('mujs-body');
  3100. this.cfgpage = make('mujs-row', 'mujs-cfg hidden');
  3101. this.main = make('mujs-main', 'hidden');
  3102. this.urlContainer = make('mujs-url');
  3103. this.btnframe = make('mujs-column', 'btn-frame');
  3104. this.btnHandles = make('mujs-column', 'btn-handles');
  3105. this.btnHide = make('mujs-btn', 'hide-list', {
  3106. title: i18n$('min'),
  3107. innerHTML: iconSVG.load('minus'),
  3108. dataset: {
  3109. command: 'hide-list'
  3110. }
  3111. });
  3112. this.btnfullscreen = make('mujs-btn', 'fullscreen', {
  3113. title: i18n$('max'),
  3114. innerHTML: iconSVG.load('expand'),
  3115. dataset: {
  3116. command: 'fullscreen'
  3117. }
  3118. });
  3119. this.closebtn = make('mujs-btn', 'close', {
  3120. title: i18n$('close'),
  3121. innerHTML: iconSVG.load('close'),
  3122. dataset: {
  3123. command: 'close'
  3124. }
  3125. });
  3126. this.btncfg = make('mujs-btn', 'settings hidden', {
  3127. title: 'Settings',
  3128. innerHTML: iconSVG.load('gear'),
  3129. dataset: {
  3130. command: 'settings'
  3131. }
  3132. });
  3133. this.btnhome = make('mujs-btn', 'github hidden', {
  3134. title: `GitHub (v${
  3135. /\d+\.\d+\.\d+|Book/.test($info.script.version)
  3136. ? $info.script.version
  3137. : $info.script.version.slice(0, 5)
  3138. })`,
  3139. innerHTML: iconSVG.load('github'),
  3140. dataset: {
  3141. command: 'open-tab',
  3142. webpage: $info.script.namespace
  3143. }
  3144. });
  3145. this.btnissue = make('mujs-btn', 'issue hidden', {
  3146. innerHTML: iconSVG.load('issue'),
  3147. title: i18n$('issue'),
  3148. dataset: {
  3149. command: 'open-tab',
  3150. webpage: $info.script.bugs ?? 'https://github.com/magicoflolis/Userscript-Plus/issues'
  3151. }
  3152. });
  3153. this.btngreasy = make('mujs-btn', 'greasy hidden', {
  3154. title: 'Greasy Fork',
  3155. innerHTML: iconSVG.load('globe'),
  3156. dataset: {
  3157. command: 'open-tab',
  3158. webpage: 'https://greasyfork.org/scripts/421603'
  3159. }
  3160. });
  3161. this.btnnav = make('mujs-btn', 'nav', {
  3162. title: 'Navigation',
  3163. innerHTML: iconSVG.load('nav'),
  3164. dataset: {
  3165. command: 'navigation'
  3166. }
  3167. });
  3168. const makeTHead = (rows = []) => {
  3169. const tr = make('tr');
  3170. for (const r of rows) {
  3171. const tparent = make('th', r.class ?? '', r);
  3172. tr.append(tparent);
  3173. }
  3174. this.tabhead.append(tr);
  3175. this.table.append(this.tabhead, this.tabbody);
  3176. };
  3177. makeTHead([
  3178. {
  3179. class: 'mujs-header-name',
  3180. textContent: i18n$('name')
  3181. },
  3182. {
  3183. textContent: i18n$('createdby')
  3184. },
  3185. {
  3186. textContent: i18n$('daily_installs')
  3187. },
  3188. {
  3189. textContent: i18n$('updated')
  3190. },
  3191. {
  3192. textContent: i18n$('install')
  3193. }
  3194. ]);
  3195. // #endregion
  3196. if (isMobile) {
  3197. dom.cl.add([this.btnHide, this.btnfullscreen, this.closebtn], 'hidden');
  3198. this.btnframe.append(
  3199. this.btnHide,
  3200. this.btnfullscreen,
  3201. this.closebtn,
  3202. this.btnhome,
  3203. this.btngreasy,
  3204. this.btnissue,
  3205. this.btncfg,
  3206. this.btnnav
  3207. );
  3208. } else {
  3209. this.btnHandles.append(this.btnHide, this.btnfullscreen, this.closebtn);
  3210. this.btnframe.append(this.btnhome, this.btngreasy, this.btnissue, this.btncfg, this.btnnav);
  3211. }
  3212. this.toolbar.append(this.btnHandles);
  3213. this.urlContainer.append(this.urlBar);
  3214. this.header.append(this.urlContainer, this.rateContainer, this.countframe, this.btnframe);
  3215. this.tbody.append(this.table, this.cfgpage);
  3216. this.main.append(this.toolbar, this.header, this.tbody, this.footer, this.promptElem);
  3217. this.mainframe.append(this.mainbtn);
  3218. this.root.append(this.mainframe, this.main);
  3219. return true;
  3220. } catch (ex) {
  3221. err(ex);
  3222. }
  3223. return false;
  3224. }
  3225. remove() {
  3226. this.hostCache.clear();
  3227. this.userjsCache.clear();
  3228. dom.remove(this.frame);
  3229. }
  3230. async save() {
  3231. this.unsaved = false;
  3232. await StorageSystem.setValue('Config', cfg);
  3233. info('Saved config:', cfg);
  3234. this.redirect();
  3235. return cfg;
  3236. }
  3237. /**
  3238. * @param { string } css - CSS to inject
  3239. * @param { string } name - Name of stylesheet
  3240. * @return { HTMLStyleElement | undefined } Style element
  3241. */
  3242. loadCSS(css, name = 'CSS') {
  3243. try {
  3244. if (typeof name !== 'string')
  3245. throw new Error('"name" must be a typeof "string"', { cause: 'loadCSS' });
  3246. if (qs(`style[data-role="${name}"]`, this.root))
  3247. return qs(`style[data-role="${name}"]`, this.root);
  3248. if (typeof css !== 'string')
  3249. throw new Error('"css" must be a typeof "string"', { cause: 'loadCSS' });
  3250. if (isBlank(css))
  3251. throw new Error(`"${name}" contains empty CSS string`, { cause: 'loadCSS' });
  3252. const parent = isEmpty(this.root.shadowRoot) ? this.root : this.root.shadowRoot;
  3253. if (isGM) {
  3254. const fn = isFN(GM.addElement)
  3255. ? GM.addElement
  3256. : isFN(GM_addElement)
  3257. ? GM_addElement
  3258. : BLANK_FN;
  3259. const sty = fn(parent, 'style', { textContent: css });
  3260. if (isElem(sty)) {
  3261. sty.dataset.insertedBy = $info.script.name;
  3262. sty.dataset.role = name;
  3263. return sty;
  3264. }
  3265. }
  3266. const sty = make('style', {
  3267. textContent: css,
  3268. dataset: {
  3269. insertedBy: $info.script.name,
  3270. role: name
  3271. }
  3272. });
  3273. parent.appendChild(sty);
  3274. return sty;
  3275. } catch (ex) {
  3276. err(ex);
  3277. }
  3278. }
  3279. checkBlacklist(str) {
  3280. str = str || this.host;
  3281. if (/accounts*\.google\./.test(this.webpage.host)) {
  3282. this.isBlacklisted = true;
  3283. return this.isBlacklisted;
  3284. }
  3285. let blacklisted = false;
  3286. for (const b of normalizeTarget(cfg.blacklist)) {
  3287. if (typeof b === 'string') {
  3288. if (b.startsWith('userjs-')) {
  3289. const r = /userjs-(\w+)/.exec(b)[1];
  3290. const biList = builtinList[r];
  3291. if (isRegExp(biList)) {
  3292. if (!biList.test(str)) continue;
  3293. blacklisted = true;
  3294. } else if (isObj(biList) && biList.host === this.host) {
  3295. blacklisted = true;
  3296. }
  3297. }
  3298. } else if (isObj(b)) {
  3299. if (!b.enabled) {
  3300. continue;
  3301. }
  3302. if (b.regex === true) {
  3303. const reg = new RegExp(b.url, b.flags);
  3304. if (!reg.test(str)) continue;
  3305. blacklisted = true;
  3306. }
  3307. if (Array.isArray(b.url)) {
  3308. for (const c of b.url) {
  3309. if (!str.includes(c)) continue;
  3310. blacklisted = true;
  3311. }
  3312. }
  3313. if (!str.includes(b.url)) continue;
  3314. blacklisted = true;
  3315. }
  3316. }
  3317. this.isBlacklisted = blacklisted;
  3318. return this.isBlacklisted;
  3319. }
  3320. setTheme() {
  3321. const theme = cfg.theme ?? DEFAULT_CONFIG.theme;
  3322. if (theme === DEFAULT_CONFIG.theme) {
  3323. return;
  3324. }
  3325. const sty = this.root.style;
  3326. for (const [k, v] of Object.entries(theme)) {
  3327. const str = `--mujs-${k}`;
  3328. const prop = sty.getPropertyValue(str);
  3329. if (isEmpty(v)) theme[k] = prop;
  3330. if (prop === v) continue;
  3331. sty.removeProperty(str);
  3332. sty.setProperty(str, v);
  3333. }
  3334. }
  3335. makePrompt(txt, dataset = {}, usePrompt = true) {
  3336. for (const elem of normalizeTarget(qsA('.prompt', this.promptElem))) {
  3337. if (elem.dataset.prompt) elem.remove();
  3338. }
  3339. const el = make('mu-js', 'prompt', {
  3340. dataset: {
  3341. prompt: txt
  3342. }
  3343. });
  3344. const elHead = make('mu-js', 'prompt-head', {
  3345. innerHTML: `${iconSVG.load('refresh')} ${txt}`
  3346. });
  3347. el.append(elHead);
  3348. if (usePrompt) {
  3349. const elPrompt = make('mu-js', 'prompt-body', { dataset });
  3350. const elYes = make('mujs-btn', 'prompt-confirm', {
  3351. innerHTML: 'Confirm',
  3352. dataset: {
  3353. command: 'prompt-confirm'
  3354. }
  3355. });
  3356. const elNo = make('mujs-btn', 'prompt-deny', {
  3357. innerHTML: 'Deny',
  3358. dataset: {
  3359. command: 'prompt-deny'
  3360. }
  3361. });
  3362. elPrompt.append(elYes, elNo);
  3363. el.append(elPrompt);
  3364. } else {
  3365. const elPrompt = make('mu-js', 'prompt-body');
  3366. const elNo = make('mujs-btn', 'prompt-deny', {
  3367. textContent: i18n$('close')
  3368. });
  3369. ael(elNo, isMobile ? 'touchend' : 'click', () => {
  3370. el.remove();
  3371. });
  3372. elPrompt.append(elNo);
  3373. el.append(elPrompt);
  3374. }
  3375. this.promptElem.append(el);
  3376. return el;
  3377. }
  3378. /**
  3379. * @template {string | Error} E
  3380. * @param {...E} ex
  3381. */
  3382. showError(...ex) {
  3383. err(...ex);
  3384. const error = make('mu-js', 'error');
  3385. let str = '';
  3386. for (const e of ex) {
  3387. str += `${typeof e === 'string' ? e : `${e.cause ? `[${e.cause}] ` : ''}${e.message}${e.stack ? ` ${e.stack}` : ''}`}\n`;
  3388. }
  3389. const { createTextNode } = _self;
  3390. error.appendChild(createTextNode(str));
  3391. this.footer.append(error);
  3392. }
  3393. refresh() {
  3394. this.urlBar.placeholder = i18n$('newTab');
  3395. Counter.reset();
  3396. dom.cl.remove(this.toElem(), 'hidden');
  3397. dom.cl.remove(cfgSec, 'hidden');
  3398. dom.prop([this.tabbody, this.rateContainer, this.footer], 'innerHTML', '');
  3399. }
  3400. /**
  3401. * Redirects sleazyfork userscripts from greasyfork.org to sleazyfork.org
  3402. *
  3403. * Taken from: https://greasyfork.org/scripts/23840
  3404. */
  3405. redirect() {
  3406. const locObj = window.top.location;
  3407. const { hostname } = locObj;
  3408. const gfSite = /greasyfork\.org/.test(hostname);
  3409. if (!gfSite && cfg.sleazyredirect) {
  3410. return;
  3411. }
  3412. const otherSite = gfSite ? 'sleazyfork' : 'greasyfork';
  3413. if (!qs('span.sign-in-link')) {
  3414. return;
  3415. }
  3416. if (!/scripts\/\d+/.test(locObj.href)) {
  3417. return;
  3418. }
  3419. if (
  3420. !qs('#script-info') &&
  3421. (otherSite == 'greasyfork' || qs('div.width-constraint>section>p>a'))
  3422. ) {
  3423. const str = locObj.href.replace(
  3424. /\/\/([^.]+\.)?(greasyfork|sleazyfork)\.org/,
  3425. '//$1' + otherSite + '.org'
  3426. );
  3427. info(`Redirecting to "${str}"`);
  3428. if (isFN(locObj.assign)) {
  3429. locObj.assign(str);
  3430. } else {
  3431. locObj.href = str;
  3432. }
  3433. }
  3434. }
  3435. /**
  3436. * @param {number} [time]
  3437. */
  3438. async timeoutFrame(time) {
  3439. const frameTimeout = this.timeouts.frame;
  3440. frameTimeout.clear(...frameTimeout.ids);
  3441. if (dom.cl.has(this.mainframe, 'hidden')) {
  3442. return;
  3443. }
  3444. time = time ?? cfg.time ?? DEFAULT_CONFIG.time;
  3445. let n = 10000;
  3446. if (typeof time === 'number' && !Number.isNaN(time)) {
  3447. n = this.isBlacklisted ? time / 2 : time;
  3448. }
  3449. await frameTimeout.set(n);
  3450. this.remove();
  3451. frameTimeout.clear(...frameTimeout.ids);
  3452. }
  3453. toElem() {
  3454. return Array.from(this).map(({ _mujs }) => {
  3455. return _mujs.root;
  3456. });
  3457. }
  3458. *[Symbol.iterator]() {
  3459. const arr = Array.from(this.userjsCache.values()).filter(({ _mujs }) => {
  3460. return !isEmpty(_mujs) && _mujs.info.engine.enabled;
  3461. });
  3462. for (const userjs of arr) {
  3463. yield userjs;
  3464. }
  3465. }
  3466. }
  3467. const container = new Container();
  3468. // #endregion
  3469. // #region Primary Function
  3470. function primaryFN() {
  3471. const respHandles = {
  3472. build: BLANK_ASYNC_FN
  3473. };
  3474. try {
  3475. const { scheduler } = _self;
  3476. const {
  3477. mainframe,
  3478. urlBar,
  3479. rateContainer,
  3480. footer,
  3481. tabbody,
  3482. cfgpage,
  3483. btnfullscreen,
  3484. main,
  3485. Tabs,
  3486. showError
  3487. } = container;
  3488. const reloadCfg = () => {
  3489. for (const base of cfgBase) {
  3490. const nm = /^(\w+)-(.+)/.exec(base.value);
  3491. const d = (() => {
  3492. if (base.tag === 'engine') {
  3493. const engine = DEFAULT_CONFIG.engines.find((engine) => engine.name === base.value);
  3494. if (engine) {
  3495. return engine;
  3496. }
  3497. }
  3498. if (nm) {
  3499. return DEFAULT_CONFIG[nm[1]][nm[2]];
  3500. }
  3501. return DEFAULT_CONFIG[base.value];
  3502. })();
  3503. const v = (() => {
  3504. if (base.tag === 'engine') {
  3505. const engine = cfg.engines.find((engine) => engine.name === base.value);
  3506. if (engine) {
  3507. return engine;
  3508. }
  3509. }
  3510. if (nm) {
  3511. return cfg[nm[1]][nm[2]];
  3512. }
  3513. return cfg[base.value];
  3514. })();
  3515. base.cache = v;
  3516. if (base.type === 'checkbox') {
  3517. if (nm) {
  3518. if (nm[1] === 'filters') {
  3519. base.elem.checked = cfg[nm[1]][nm[2]].enabled;
  3520. } else {
  3521. base.elem.checked = v;
  3522. }
  3523. } else if (base.tag === 'engine') {
  3524. base.elem.checked = v.enabled;
  3525. base.elemUrl.value = decode(v.query);
  3526. base.elemUrl.placeholder = decode(d.query);
  3527. if (base.elemToken) {
  3528. base.elemToken = v.token;
  3529. }
  3530. }
  3531. } else {
  3532. base.elem.value = v;
  3533. }
  3534. }
  3535. container.setTheme();
  3536. };
  3537. const doInstallProcess = async (installLink) => {
  3538. const locObj = window.top.location;
  3539. if (isFN(locObj.assign)) {
  3540. locObj.assign(installLink.href);
  3541. } else {
  3542. locObj.href = installLink.href;
  3543. }
  3544. installLink.remove();
  3545. await init();
  3546. };
  3547. const doDownloadProcess = async (details) => {
  3548. if (!details.url) {
  3549. return;
  3550. }
  3551. const a = make('a');
  3552. a.href = details.url;
  3553. a.setAttribute('download', details.filename || '');
  3554. a.setAttribute('type', 'text/plain');
  3555. a.dispatchEvent(new MouseEvent('click'));
  3556. await init();
  3557. };
  3558. const applyTo = (ujs, name, elem, root) => {
  3559. const n = ujs._mujs.code[name] ?? ujs._mujs.code.data_meta[name];
  3560. if (isEmpty(n)) {
  3561. const el = make('mujs-a', {
  3562. textContent: i18n$('listing_none')
  3563. });
  3564. elem.append(el);
  3565. return;
  3566. }
  3567. dom.prop(elem, 'innerHTML', '');
  3568. dom.cl.remove(root, 'hidden');
  3569. if (isObj(n)) {
  3570. if (name === 'resource') {
  3571. for (const [k, v] of Object.entries(n)) {
  3572. const el = make('mujs-a', {
  3573. textContent: k ?? 'ERROR'
  3574. });
  3575. if (v.startsWith('http')) {
  3576. el.dataset.command = 'open-tab';
  3577. el.dataset.webpage = v;
  3578. }
  3579. elem.append(el);
  3580. }
  3581. } else {
  3582. const el = make('mujs-a', {
  3583. textContent: n.text
  3584. });
  3585. if (n.domain) {
  3586. el.dataset.command = 'open-tab';
  3587. el.dataset.webpage = `https://${n.text}`;
  3588. }
  3589. elem.append(el);
  3590. }
  3591. } else if (typeof n === 'string') {
  3592. const el = make('mujs-a', {
  3593. textContent: n
  3594. });
  3595. elem.append(el);
  3596. } else {
  3597. for (const c of n) {
  3598. if (typeof c === 'string' && c.startsWith('http')) {
  3599. const el = make('mujs-a', {
  3600. textContent: c,
  3601. dataset: {
  3602. command: 'open-tab',
  3603. webpage: c
  3604. }
  3605. });
  3606. elem.append(el);
  3607. } else if (isObj(c)) {
  3608. const el = make('mujs-a', {
  3609. textContent: c.text
  3610. });
  3611. if (c.domain) {
  3612. el.dataset.command = 'open-tab';
  3613. el.dataset.webpage = `https://${c.text}`;
  3614. }
  3615. elem.append(el);
  3616. } else {
  3617. const el = make('mujs-a', {
  3618. textContent: c
  3619. });
  3620. elem.append(el);
  3621. }
  3622. }
  3623. }
  3624. };
  3625. // #region Main event handlers
  3626. const frameTimeout = container.timeouts.frame;
  3627. ael(main, isMobile ? 'touchend' : 'click', async (evt) => {
  3628. try {
  3629. /** @type { HTMLElement } */
  3630. const target = evt.target.closest('[data-command]');
  3631. if (!target) {
  3632. return;
  3633. }
  3634. let dataset = target.dataset;
  3635. let cmd = dataset.command;
  3636. if (/^prompt-/.test(target.dataset.command)) {
  3637. dataset = target.parentElement.dataset;
  3638. cmd = dataset.command;
  3639. let pElem = target.parentElement.parentElement;
  3640. if (/prompt-install/.test(target.dataset.command)) {
  3641. pElem = target.parentElement.parentElement.parentElement;
  3642. const a = make('a', {
  3643. onclick(evt) {
  3644. evt.preventDefault();
  3645. doInstallProcess(evt.target);
  3646. }
  3647. });
  3648. a.href = target.dataset.code_url;
  3649. a.click();
  3650. } else if (/prompt-download/.test(target.dataset.command)) {
  3651. pElem = target.parentElement.parentElement.parentElement;
  3652. const dataUserJS = container.userjsCache.get(+target.dataset.userjs);
  3653. if (dataUserJS) {
  3654. const code_obj = await dataUserJS._mujs.code.request(false, target.dataset.code_url);
  3655. if (typeof code_obj.code === 'string')
  3656. doDownloadProcess({
  3657. url: 'data:text/plain;charset=utf-8,' + encodeURIComponent(code_obj.code),
  3658. filename: `${dataUserJS.name}${isUserCSS(target.dataset.code_url) ? '.user.css' : '.user.js'}`
  3659. });
  3660. }
  3661. }
  3662. pElem.remove();
  3663. return;
  3664. }
  3665. if (cmd === 'install-script') {
  3666. const dataUserJS = container.userjsCache.get(+dataset.userjs);
  3667. if (isNull(dataUserJS)) {
  3668. return;
  3669. }
  3670. if (dataUserJS.code_urls.length > 1) {
  3671. const list = make('mujs-list', {
  3672. style: 'display: flex; flex-direction: column;'
  3673. });
  3674. for (const ujs of dataUserJS.code_urls) {
  3675. const a = make('mujs-a', {
  3676. title: ujs.code_url,
  3677. textContent: ujs.name,
  3678. dataset: {
  3679. command: 'prompt-install',
  3680. code_url: ujs.code_url
  3681. }
  3682. });
  3683. list.append(a);
  3684. }
  3685. container.makePrompt(`Multiple detected: ${list.outerHTML}`, dataset, false);
  3686. } else {
  3687. const a = make('a', {
  3688. onclick(evt) {
  3689. evt.preventDefault();
  3690. doInstallProcess(evt.target);
  3691. }
  3692. });
  3693. a.href = dataUserJS.code_url;
  3694. a.click();
  3695. }
  3696. } else if (cmd === 'open-tab' && dataset.webpage) {
  3697. if (isGM) {
  3698. if (isFN(GM.openInTab)) {
  3699. return GM.openInTab(dataset.webpage);
  3700. } else if (isFN(GM_openInTab)) {
  3701. return GM_openInTab(dataset.webpage, {
  3702. active: true,
  3703. insert: true
  3704. });
  3705. }
  3706. }
  3707. return window.open(dataset.webpage, '_blank');
  3708. } else if (cmd === 'navigation') {
  3709. for (const e of normalizeTarget(qsA('mujs-btn', target.parentElement)).filter(
  3710. (e) => !dom.cl.has(e, 'nav')
  3711. )) {
  3712. dom.cl.toggle(e, 'hidden');
  3713. }
  3714. } else if (cmd === 'list-description') {
  3715. const arr = [];
  3716. const ignoreTags = new Set(['TD', 'MUJS-A', 'MU-JS']);
  3717. for (const node of Object.values(target.parentElement._mujs)) {
  3718. if (ignoreTags.has(node.tagName)) {
  3719. continue;
  3720. }
  3721. if (node.tagName === 'TEXTAREA' && isEmpty(node.value)) {
  3722. continue;
  3723. }
  3724. arr.push(node);
  3725. }
  3726. if (target.nextElementSibling) {
  3727. arr.push(target.nextElementSibling);
  3728. if (target.nextElementSibling.nextElementSibling) {
  3729. arr.push(target.nextElementSibling.nextElementSibling);
  3730. }
  3731. }
  3732. if (dom.cl.has(arr[0], 'hidden')) {
  3733. dom.cl.remove(arr, 'hidden');
  3734. } else {
  3735. dom.cl.add(arr, 'hidden');
  3736. }
  3737. } else if (cmd === 'close') {
  3738. container.remove();
  3739. } else if (cmd === 'fullscreen') {
  3740. if (dom.cl.has(btnfullscreen, 'expanded')) {
  3741. dom.cl.remove([btnfullscreen, main], 'expanded');
  3742. dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('expand'));
  3743. } else {
  3744. dom.cl.add([btnfullscreen, main], 'expanded');
  3745. dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('collapse'));
  3746. }
  3747. } else if (cmd === 'hide-list') {
  3748. dom.cl.add(main, 'hidden');
  3749. dom.cl.remove(mainframe, 'hidden');
  3750. container.timeoutFrame();
  3751. } else if (cmd === 'save') {
  3752. container.rebuild = true;
  3753. dom.prop(rateContainer, 'innerHTML', '');
  3754. if (!dom.prop(target, 'disabled')) {
  3755. const config = await container.save();
  3756. if (container.rebuild) {
  3757. if (config.autofetch) {
  3758. respHandles.build();
  3759. }
  3760. }
  3761. container.unsaved = false;
  3762. container.rebuild = false;
  3763. }
  3764. } else if (cmd === 'reset') {
  3765. cfg = DEFAULT_CONFIG;
  3766. dom.remove(qsA('.error', footer));
  3767. container.unsaved = true;
  3768. container.rebuild = true;
  3769. reloadCfg();
  3770. } else if (cmd === 'settings') {
  3771. if (container.unsaved) {
  3772. showError('Unsaved changes');
  3773. }
  3774. Tabs.create('mujs:settings');
  3775. container.rebuild = false;
  3776. } else if (cmd === 'new-tab') {
  3777. Tabs.create();
  3778. } else if (cmd === 'switch-tab') {
  3779. Tabs.active(target);
  3780. } else if (cmd === 'close-tab' && target.parentElement) {
  3781. Tabs.close(target.parentElement);
  3782. } else if (cmd === 'download-userjs') {
  3783. const dataUserJS = container.userjsCache.get(+dataset.userjs);
  3784. if (isNull(dataUserJS)) {
  3785. return;
  3786. }
  3787. if (dataUserJS.code_urls.length > 1) {
  3788. const list = make('mujs-list', {
  3789. style: 'display: flex; flex-direction: column;'
  3790. });
  3791. for (const ujs of dataUserJS.code_urls) {
  3792. const a = make('mujs-a', {
  3793. title: ujs.code_url,
  3794. textContent: ujs.name,
  3795. dataset: {
  3796. command: 'prompt-download',
  3797. code_url: ujs.code_url,
  3798. userjs: dataset.userjs
  3799. }
  3800. });
  3801. list.append(a);
  3802. }
  3803. container.makePrompt(`Multiple detected: ${list.outerHTML}`, dataset, false);
  3804. } else {
  3805. const code_obj = await dataUserJS._mujs.code.request(false);
  3806. if (typeof code_obj.code === 'string')
  3807. doDownloadProcess({
  3808. url: 'data:text/plain;charset=utf-8,' + encodeURIComponent(code_obj.code),
  3809. filename: `${dataUserJS.name}${isUserCSS(dataUserJS.code_url) ? '.user.css' : '.user.js'}`
  3810. });
  3811. }
  3812. } else if (cmd === 'load-userjs' || cmd === 'load-header') {
  3813. if (!container.userjsCache.has(+dataset.userjs)) {
  3814. return;
  3815. }
  3816. const codeArea = qs('textarea', target.parentElement.parentElement);
  3817. if (!isEmpty(codeArea.value) && cmd === codeArea.dataset.load) {
  3818. dom.cl.toggle(codeArea, 'hidden');
  3819. return;
  3820. }
  3821. codeArea.dataset.load = cmd;
  3822. const dataUserJS = container.userjsCache.get(+dataset.userjs);
  3823. const code_obj = await dataUserJS._mujs.code.request();
  3824. if (typeof code_obj.data_code_block !== 'string') {
  3825. codeArea.value = 'An error occured';
  3826. return;
  3827. }
  3828. codeArea.value =
  3829. cmd === 'load-userjs' ? code_obj.data_code_block : code_obj.data_meta_block;
  3830. dom.cl.remove(codeArea, 'hidden');
  3831. for (const e of qsA(
  3832. 'mujs-column[data-el="matches"]',
  3833. target.parentElement.parentElement
  3834. )) {
  3835. applyTo(dataUserJS, e.dataset.type, qs('.mujs-grants', e), e);
  3836. }
  3837. } else if (cmd === 'load-page') {
  3838. if (!container.userjsCache.has(+dataset.userjs)) {
  3839. return;
  3840. }
  3841. let pageArea = qs('mujs-page', target.parentElement.parentElement);
  3842. if (!pageArea) {
  3843. pageArea = make('mujs-page');
  3844. target.parentElement.parentElement.append(pageArea);
  3845. const dataUserJS = container.userjsCache.get(+dataset.userjs);
  3846. const engine = dataUserJS._mujs.info.engine;
  3847. let pageURL;
  3848. if (engine.name.includes('fork')) {
  3849. const { navigator } = _self;
  3850. const current = navigator.language.split('-')[0] ?? 'en';
  3851. pageURL = dataUserJS.url.replace(
  3852. /\/scripts/,
  3853. `/${/^(zh|fr|es)/.test(current) ? navigator.language : current}/scripts`
  3854. );
  3855. } else if (engine.name.includes('github')) {
  3856. const page_url = await Network.req(dataUserJS.page_url, 'GET', 'json', {
  3857. headers: {
  3858. Accept: 'application/vnd.github+json',
  3859. Authorization: `Bearer ${engine.token}`,
  3860. 'X-GitHub-Api-Version': '2022-11-28'
  3861. }
  3862. }).catch(() => {
  3863. return {};
  3864. });
  3865. if (!page_url.download_url) {
  3866. return;
  3867. }
  3868. const page = await Network.req(page_url.download_url, 'GET', 'text');
  3869. if (container.supported) {
  3870. const shadow = pageArea.attachShadow({ mode: 'closed' });
  3871. const div = make('div', {
  3872. innerHTML: page
  3873. });
  3874. shadow.append(div);
  3875. }
  3876. return;
  3877. } else {
  3878. pageURL = dataUserJS.url;
  3879. }
  3880. if (!pageURL) {
  3881. return;
  3882. }
  3883. const page = await Network.req(pageURL, 'GET', 'document');
  3884. const getContent = () => {
  3885. let content = 'An error occured';
  3886. const h = new URL(dataUserJS.url);
  3887. const root = qs('.user-content', page.documentElement);
  3888. for (const e of qsA('[href]', root)) {
  3889. e.target = '_blank';
  3890. e.style = 'pointer-events: auto;';
  3891. if (e.href.startsWith('/')) {
  3892. e.href = `${h.origin}${e.href}`;
  3893. }
  3894. }
  3895. for (const e of qsA('img[src]', root)) {
  3896. e.style =
  3897. 'max-width: 25em; max-height: 25em; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none;';
  3898. }
  3899. if (root) {
  3900. content = root.innerHTML;
  3901. } else {
  3902. content = 'No additional info available';
  3903. }
  3904. return content;
  3905. };
  3906. if (container.supported) {
  3907. const shadow = pageArea.attachShadow({ mode: 'closed' });
  3908. const div = make('div', {
  3909. style: 'pointer-events: none;',
  3910. innerHTML: getContent()
  3911. });
  3912. shadow.append(div);
  3913. }
  3914. return;
  3915. }
  3916. if (!dom.cl.has(pageArea, 'hidden')) {
  3917. dom.cl.add(pageArea, 'hidden');
  3918. return;
  3919. }
  3920. dom.cl.remove(pageArea, 'hidden');
  3921. } else if (/export-/.test(cmd)) {
  3922. const toCfg = cmd === 'export-cfg';
  3923. doDownloadProcess({
  3924. url:
  3925. 'data:text/plain;charset=utf-8,' +
  3926. encodeURIComponent(JSON.stringify(toCfg ? cfg : cfg.theme, null, ' ')),
  3927. filename: `Magic_Userscript_${toCfg ? 'config' : 'theme'}.json`
  3928. });
  3929. } else if (/import-/.test(cmd)) {
  3930. if (qs('input', target.parentElement)) {
  3931. qs('input', target.parentElement).click();
  3932. return;
  3933. }
  3934. const inpJSON = make('input', 'hidden', {
  3935. type: 'file',
  3936. accept: '.json',
  3937. onchange(evt) {
  3938. const file = evt.target.files[0];
  3939. if (file === undefined || file.name === '') {
  3940. return;
  3941. }
  3942. const fr = new FileReader();
  3943. fr.onload = function () {
  3944. if (typeof fr.r###lt !== 'string') {
  3945. return;
  3946. }
  3947. const content = JSON.parse(fr.r###lt);
  3948. if (content.blacklist) {
  3949. cfg = content;
  3950. container.unsaved = true;
  3951. container.rebuild = true;
  3952. reloadCfg();
  3953. container.save().then((config) => {
  3954. if (config.autofetch) {
  3955. respHandles.build();
  3956. }
  3957. container.unsaved = false;
  3958. container.rebuild = false;
  3959. });
  3960. } else {
  3961. cfg.theme = content;
  3962. container.setTheme();
  3963. }
  3964. inpJSON.remove();
  3965. };
  3966. fr.readAsText(file);
  3967. }
  3968. });
  3969. target.parentElement.append(inpJSON);
  3970. inpJSON.click();
  3971. }
  3972. } catch (ex) {
  3973. showError(ex);
  3974. }
  3975. });
  3976. ael(main, 'auxclick', (evt) => {
  3977. if (evt.button !== 1) {
  3978. return;
  3979. }
  3980. /** @type { HTMLElement } */
  3981. const target = evt.target.closest('[data-command]');
  3982. if (!target) {
  3983. return;
  3984. }
  3985. const dataset = target.dataset;
  3986. const cmd = dataset.command;
  3987. if (cmd === 'switch-tab' || cmd === 'close-tab') {
  3988. Tabs.close(target);
  3989. } else if (cmd === 'new-tab') {
  3990. Tabs.create();
  3991. }
  3992. });
  3993. if (!isMobile) {
  3994. const fade = async (target, type) => {
  3995. if (type === 'mouseenter') {
  3996. frameTimeout.clear(...frameTimeout.ids);
  3997. container.timeouts.mouse.clear(...container.timeouts.mouse.ids);
  3998. target.style.opacity = container.opacityMax;
  3999. } else if (type === 'mouseleave') {
  4000. await container.timeouts.mouse.set(cfg.time);
  4001. target.style.opacity = container.opacityMin;
  4002. }
  4003. };
  4004. for (const e of ['mouseenter', 'mouseleave']) {
  4005. ael(main, e, (evt) => {
  4006. evt.preventDefault();
  4007. evt.stopPropagation();
  4008. fade(evt.target, evt.type);
  4009. });
  4010. }
  4011. }
  4012. /**
  4013. * @param {CustomEvent<import("../typings/types.d.ts").GSForkQuery>} evt
  4014. */
  4015. const updatedItem = (evt) => {
  4016. const ujs = evt.detail;
  4017. if (!ujs._mujs) return;
  4018. if (ujs.deleted === true) {
  4019. ujs._mujs.root.remove();
  4020. container.userjsCache.delete(ujs.id);
  4021. Counter.reset();
  4022. MUList.sortRecords();
  4023. return;
  4024. }
  4025. if (!isEmpty(ujs.code_urls)) ujs.code_url = ujs.code_urls[0].code_url;
  4026. for (const elem of qsA('[data-name]', ujs._mujs.root)) {
  4027. const name = elem.dataset.name;
  4028. if (name === 'code') {
  4029. if (ujs._mujs.code.data_code_block) {
  4030. if (cfg.preview.code && !cfg.preview.metadata) {
  4031. elem.value = ujs._mujs.code.data_code_block;
  4032. } else if (cfg.preview.metadata && !cfg.preview.code) {
  4033. elem.value = ujs._mujs.code.data_meta_block;
  4034. } else {
  4035. elem.value = `${ujs._mujs.code.META_START_COMMENT}${ujs._mujs.code.data_meta_block}${ujs._mujs.code.META_END_COMMENT}${ujs._mujs.code.data_code_block}`;
  4036. }
  4037. }
  4038. continue;
  4039. }
  4040. if (!ujs[name]) continue;
  4041. if (name === 'license') {
  4042. dom.attr(elem, 'title', ujs.license ?? i18n$('no_license'));
  4043. dom.text(elem, `${i18n$('license')}: ${ujs.license ?? i18n$('no_license')}`);
  4044. } else if (name === 'code_updated_at') {
  4045. dom.text(elem, language.toDate(ujs.code_updated_at));
  4046. elem.dataset.value = new Date(ujs.code_updated_at).toISOString();
  4047. } else if (name === 'created_date') {
  4048. dom.text(elem, `${i18n$('created_date')}: ${language.toDate(ujs.created_at)}`);
  4049. elem.dataset.value = new Date(ujs.created_at).toISOString();
  4050. } else if (name === 'total_installs') {
  4051. dom.text(elem, `${i18n$('total_installs')}: ${language.toNumber(ujs.total_installs)}`);
  4052. } else {
  4053. dom.text(elem, ujs[name]);
  4054. }
  4055. }
  4056. if (ujs._mujs.code.data_code_block) {
  4057. for (const e of qsA('mujs-column[data-el="matches"]', ujs._mujs.root)) {
  4058. applyTo(ujs, e.dataset.type, qs('.mujs-grants', e), e);
  4059. }
  4060. }
  4061. if (container.userjsCache.has(ujs.id)) container.userjsCache.set(ujs.id, ujs);
  4062. };
  4063. ael(main, 'updateditem', updatedItem);
  4064. // #endregion
  4065. const TLD_EXPANSION = ['com', 'net', 'org', 'de', 'co.uk'];
  4066. const APPLIES_TO_ALL_PATTERNS = [
  4067. 'http://*',
  4068. 'https://*',
  4069. 'http://*/*',
  4070. 'https://*/*',
  4071. 'http*://*',
  4072. 'http*://*/*',
  4073. '*',
  4074. '*://*',
  4075. '*://*/*',
  4076. 'http*'
  4077. ];
  4078. class ParseUserJS {
  4079. /**
  4080. * @type { string }
  4081. */
  4082. code;
  4083. /**
  4084. * @type { string }
  4085. */
  4086. data_meta_block;
  4087. /**
  4088. * @type { string }
  4089. */
  4090. data_code_block;
  4091. /**
  4092. * @type { { [meta: string]: string | string[] | { [resource: string]: string } } }
  4093. */
  4094. data_meta;
  4095. /**
  4096. * @type { {text: string;domain: boolean;tld_extra: boolean}[] }
  4097. */
  4098. data_names;
  4099. constructor(code, isCSS) {
  4100. this.isUserCSS = isCSS === true;
  4101. this.META_START_COMMENT = this.isUserCSS ? '/* ==UserStyle==' : '// ==UserScript==';
  4102. this.META_END_COMMENT = this.isUserCSS ? '==/UserStyle== */' : '// ==/UserScript==';
  4103. if (code) {
  4104. this.code = code;
  4105. this.get_meta_block();
  4106. this.get_code_block();
  4107. this.parse_meta();
  4108. this.calculate_applies_to_names();
  4109. }
  4110. }
  4111. get_meta_block() {
  4112. if (isEmpty(this.code)) {
  4113. return '';
  4114. }
  4115. if (this.data_meta_block) {
  4116. return this.data_meta_block;
  4117. }
  4118. const start_block = this.code.indexOf(this.META_START_COMMENT);
  4119. if (isNull(start_block)) {
  4120. return '';
  4121. }
  4122. const end_block = this.code.indexOf(this.META_END_COMMENT, start_block);
  4123. if (isNull(end_block)) {
  4124. return '';
  4125. }
  4126. const meta_block = this.code.substring(
  4127. start_block + this.META_START_COMMENT.length,
  4128. end_block
  4129. );
  4130. this.data_meta_block = meta_block;
  4131. return this.data_meta_block;
  4132. }
  4133. get_code_block() {
  4134. if (isEmpty(this.code)) {
  4135. return '';
  4136. }
  4137. if (this.data_code_block) {
  4138. return this.data_code_block;
  4139. }
  4140. const start_block = this.code.indexOf(this.META_START_COMMENT);
  4141. if (isNull(start_block)) {
  4142. return null;
  4143. }
  4144. const end_block = this.code.indexOf(this.META_END_COMMENT, start_block);
  4145. if (isNull(end_block)) {
  4146. return null;
  4147. }
  4148. const code_block = this.code.substring(
  4149. end_block + this.META_END_COMMENT.length,
  4150. this.code.length
  4151. );
  4152. this.data_code_block = code_block.split('\n').filter(Boolean).join('\n');
  4153. return this.data_code_block;
  4154. }
  4155. parse_meta() {
  4156. if (isEmpty(this.code)) {
  4157. return {};
  4158. }
  4159. if (this.data_meta) {
  4160. return this.data_meta;
  4161. }
  4162. const meta = {};
  4163. const meta_block_map = new Map();
  4164. for (const meta_line of this.get_meta_block().split('\n').filter(Boolean)) {
  4165. const meta_match = /\/\/\s+@([a-zA-Z:-]+)\s+(.*)/.exec(meta_line);
  4166. if (!meta_match) continue;
  4167. const key = meta_match[1].trim();
  4168. const value = meta_match[2].trim();
  4169. if (!meta_block_map.has(key)) meta_block_map.set(key, []);
  4170. const meta_map = meta_block_map.get(key);
  4171. meta_map.push(value);
  4172. meta_block_map.set(key, meta_map);
  4173. }
  4174. for (const [key, value] of meta_block_map) {
  4175. if (value.length > 1) {
  4176. meta[key] = value;
  4177. } else {
  4178. meta[key] = value[0];
  4179. }
  4180. }
  4181. this.data_meta = meta;
  4182. return this.data_meta;
  4183. }
  4184. calculate_applies_to_names() {
  4185. if (isEmpty(this.code)) {
  4186. return [];
  4187. }
  4188. if (this.data_names) {
  4189. return this.data_names;
  4190. }
  4191. let patterns = [];
  4192. for (const [k, v] of Object.entries(this.parse_meta())) {
  4193. if (/include|match/i.test(k)) {
  4194. if (Array.isArray(v)) {
  4195. patterns = patterns.concat(v);
  4196. } else {
  4197. patterns = patterns.concat([v]);
  4198. }
  4199. }
  4200. }
  4201. if (isEmpty(patterns)) {
  4202. return [];
  4203. }
  4204. if (this.intersect(patterns, APPLIES_TO_ALL_PATTERNS)) {
  4205. this.data_names = [
  4206. {
  4207. domain: false,
  4208. text: 'All sites',
  4209. tld_extra: false
  4210. }
  4211. ];
  4212. return this.data_names;
  4213. }
  4214. this.data_names = ParseUserJS.getNames(patterns);
  4215. return this.data_names;
  4216. }
  4217. intersect(a, ...arr) {
  4218. return !isBlank([...new Set(a)].filter((v) => arr.every((b) => b.includes(v))));
  4219. }
  4220. static getNames(patterns = []) {
  4221. const name_map = new Map();
  4222. const addObj = (obj) => {
  4223. if (name_map.has(obj.text)) {
  4224. return;
  4225. }
  4226. name_map.set(obj.text, obj);
  4227. };
  4228. for (let p of patterns) {
  4229. const original_pattern = p;
  4230. let pre_wildcards = [];
  4231. if (p.match(/^\/(.*)\/$/)) {
  4232. pre_wildcards = [p];
  4233. } else {
  4234. let m = /^\*(https?:.*)/i.exec(p);
  4235. if (m) {
  4236. p = m[1];
  4237. }
  4238. p = p
  4239. .replace(/^\*:/i, 'http:')
  4240. .replace(/^\*\/\//i, 'http://')
  4241. .replace(/^http\*:/i, 'http:')
  4242. .replace(/^(https?):([^/])/i, '$1://$2');
  4243. m = /^([a-z]+:\/\/)\*\.?([a-z0-9-]+(?:.[a-z0-9-]+)+.*)/i.exec(p);
  4244. if (m) {
  4245. p = m[1] + m[2];
  4246. }
  4247. m = /^\*\.?([a-z0-9-]+\.[a-z0-9-]+.*)/i.exec(p);
  4248. if (m) {
  4249. p = `http://${m[1]}`;
  4250. }
  4251. m = /^http\*(?:\/\/)?\.?((?:[a-z0-9-]+)(?:\.[a-z0-9-]+)+.*)/i.exec(p);
  4252. if (m) {
  4253. p = `http://${m[1]}`;
  4254. }
  4255. m = /^([a-z]+:\/\/([a-z0-9-]+(?:\.[a-z0-9-]+)*\.))\*(.*)/.exec(p);
  4256. if (m) {
  4257. if (m[2].match(/A([0-9]+\.){2,}z/)) {
  4258. p = `${m[1]}tld${m[3]}`;
  4259. pre_wildcards = [p.split('*')[0]];
  4260. } else {
  4261. pre_wildcards = [p];
  4262. }
  4263. } else {
  4264. pre_wildcards = [p];
  4265. }
  4266. }
  4267. for (const pre_wildcard of pre_wildcards) {
  4268. try {
  4269. const urlObj = new URL(pre_wildcard);
  4270. const { host } = urlObj;
  4271. if (isNull(host)) {
  4272. addObj({ text: original_pattern, domain: false, tld_extra: false });
  4273. } else if (!host.includes('.') && host.includes('*')) {
  4274. addObj({ text: original_pattern, domain: false, tld_extra: false });
  4275. } else if (host.endsWith('.tld')) {
  4276. for (let i = 0; i < TLD_EXPANSION.length; i++) {
  4277. const tld = TLD_EXPANSION[i];
  4278. addObj({
  4279. text: host.replace(/tld$/i, tld),
  4280. domain: true,
  4281. tld_extra: i != 0
  4282. });
  4283. }
  4284. } else if (host.endsWith('.')) {
  4285. addObj({
  4286. text: host.slice(0, -1),
  4287. domain: true,
  4288. tld_extra: false
  4289. });
  4290. } else {
  4291. addObj({
  4292. text: host,
  4293. domain: true,
  4294. tld_extra: false
  4295. });
  4296. }
  4297. } catch {
  4298. addObj({ text: original_pattern, domain: false, tld_extra: false });
  4299. }
  4300. }
  4301. }
  4302. return [...name_map.values()];
  4303. }
  4304. /**
  4305. * @template {import("../typings/types.d.ts").GSForkQuery} O
  4306. * @param {boolean} translate
  4307. * @param {O["code_url"]} code_url
  4308. * @param {O} [obj]
  4309. */
  4310. async request(translate = false, code_url, obj) {
  4311. if (this.data_code_block) {
  4312. return this;
  4313. }
  4314. /** @type { string } */
  4315. const code = await Network.req(code_url, 'GET', 'text').catch(err);
  4316. if (typeof code !== 'string') {
  4317. return this;
  4318. }
  4319. this.isUserCSS = isUserCSS(code_url);
  4320. this.META_START_COMMENT = this.isUserCSS ? '/* ==UserStyle==' : '// ==UserScript==';
  4321. this.META_END_COMMENT = this.isUserCSS ? '==/UserStyle== */' : '// ==/UserScript==';
  4322. this.code = code;
  4323. this.get_meta_block();
  4324. this.get_code_block();
  4325. this.parse_meta();
  4326. this.calculate_applies_to_names();
  4327. const { data_meta } = this;
  4328. if (translate) {
  4329. if (data_meta[`name:${language.current}`]) {
  4330. Object.assign(obj, {
  4331. name: data_meta[`name:${language.current}`]
  4332. });
  4333. this.translated = true;
  4334. }
  4335. if (data_meta[`description:${language.current}`]) {
  4336. Object.assign(obj, {
  4337. description: data_meta[`description:${language.current}`]
  4338. });
  4339. this.translated = true;
  4340. }
  4341. }
  4342. if (Array.isArray(data_meta.grant)) {
  4343. data_meta.grant = union(data_meta.grant);
  4344. }
  4345. if (data_meta.resource) {
  4346. const obj = {};
  4347. if (typeof data_meta.resource === 'string') {
  4348. const reg = /(.+)\s+(.+)/.exec(data_meta.resource);
  4349. if (reg) {
  4350. obj[reg[1].trim()] = reg[2];
  4351. }
  4352. } else {
  4353. for (const r of data_meta.resource) {
  4354. const reg = /(.+)\s+(http.+)/.exec(r);
  4355. if (reg) {
  4356. obj[reg[1].trim()] = reg[2];
  4357. }
  4358. }
  4359. }
  4360. data_meta.resource = obj;
  4361. }
  4362. Object.assign(this, {
  4363. code_size: [Network.format(code.length)],
  4364. meta: data_meta
  4365. });
  4366. return this;
  4367. }
  4368. }
  4369. const template = {
  4370. id: 0,
  4371. bad_ratings: 0,
  4372. good_ratings: 0,
  4373. ok_ratings: 0,
  4374. daily_installs: 0,
  4375. total_installs: 0,
  4376. name: 'NOT FOUND',
  4377. description: 'NOT FOUND',
  4378. version: '0.0.0',
  4379. url: BLANK_PAGE,
  4380. code_url: BLANK_PAGE,
  4381. created_at: Date.now(),
  4382. code_updated_at: Date.now(),
  4383. locale: 'NOT FOUND',
  4384. deleted: false,
  4385. users: []
  4386. };
  4387. const mkList = (txt = '', obj = {}) => {
  4388. if (!obj.root || !obj.type) return;
  4389. const { root, type } = obj;
  4390. const appliesTo = make('mu-js', 'mujs-list', {
  4391. textContent: `${txt}: `
  4392. });
  4393. const applyList = make('mu-js', 'mujs-grants');
  4394. const ujsURLs = make('mujs-column', 'mujs-list', {
  4395. dataset: {
  4396. el: 'matches',
  4397. type
  4398. }
  4399. });
  4400. ujsURLs.append(appliesTo, applyList);
  4401. root.append(ujsURLs);
  4402. const list = obj.list ?? [];
  4403. if (isEmpty(list)) {
  4404. const elem = make('mujs-a', {
  4405. textContent: i18n$('listing_none')
  4406. });
  4407. applyList.append(elem);
  4408. dom.cl.add(ujsURLs, 'hidden');
  4409. return;
  4410. }
  4411. for (const c of list) {
  4412. if (typeof c === 'string' && c.startsWith('http')) {
  4413. const elem = make('mujs-a', {
  4414. textContent: c,
  4415. dataset: {
  4416. command: 'open-tab',
  4417. webpage: c
  4418. }
  4419. });
  4420. applyList.append(elem);
  4421. } else if (isObj(c)) {
  4422. if (type === 'resource') {
  4423. for (const [k, v] of Object.entries(c)) {
  4424. const elem = make('mujs-a', {
  4425. textContent: k ?? 'ERROR'
  4426. });
  4427. if (v.startsWith('http')) {
  4428. elem.dataset.command = 'open-tab';
  4429. elem.dataset.webpage = v;
  4430. }
  4431. applyList.append(elem);
  4432. }
  4433. } else {
  4434. const elem = make('mujs-a', {
  4435. textContent: c.text
  4436. });
  4437. if (c.domain) {
  4438. elem.dataset.command = 'open-tab';
  4439. elem.dataset.webpage = `https://${c.text}`;
  4440. }
  4441. applyList.append(elem);
  4442. }
  4443. } else {
  4444. const elem = make('mujs-a', {
  4445. textContent: c
  4446. });
  4447. applyList.append(elem);
  4448. }
  4449. }
  4450. };
  4451. // #region Create UserJS
  4452. /**
  4453. * @param { import("../typings/types.d.ts").GSForkQuery } ujs
  4454. * @param { string } engine
  4455. */
  4456. const createjs = (ujs, engine) => {
  4457. const a = [
  4458. ujs.deleted === true,
  4459. ujs.id === 421603, // Lets not add this UserJS to the list
  4460. badUserJS.includes(ujs.id),
  4461. badUserJS.includes(ujs.url)
  4462. ].some((t) => t === true);
  4463. if (a) {
  4464. return;
  4465. }
  4466. if (!container.userjsCache.has(ujs.id)) container.userjsCache.set(ujs.id, ujs);
  4467. const eframe = make('td', 'install-btn');
  4468. const uframe = make('td', 'mujs-uframe');
  4469. const fdaily = make('td', 'mujs-list', {
  4470. textContent: ujs.daily_installs,
  4471. dataset: {
  4472. name: 'daily_installs'
  4473. }
  4474. });
  4475. const fupdated = make('td', 'mujs-list', {
  4476. textContent: language.toDate(ujs.code_updated_at),
  4477. dataset: {
  4478. name: 'code_updated_at',
  4479. value: new Date(ujs.code_updated_at).toISOString()
  4480. }
  4481. });
  4482. const fname = make('td', 'mujs-name');
  4483. const fmore = make('mujs-column', 'mujs-list hidden', {
  4484. dataset: {
  4485. el: 'more-info'
  4486. }
  4487. });
  4488. const fBtns = make('mujs-column', 'mujs-list hidden');
  4489. const jsInfo = make('mujs-row', 'mujs-list');
  4490. const jsInfoB = make('mujs-row', 'mujs-list');
  4491. const ratings = make('mujs-column', 'mujs-list');
  4492. const ftitle = make('mujs-a', 'mujs-homepage', {
  4493. textContent: ujs.name,
  4494. title: ujs.url,
  4495. dataset: {
  4496. command: 'open-tab',
  4497. webpage: ujs.url
  4498. }
  4499. });
  4500. const fver = make('mu-js', 'mujs-list', {
  4501. textContent: `${i18n$('version_number')}: ${ujs.version}`
  4502. });
  4503. const fcreated = make('mu-js', 'mujs-list', {
  4504. textContent: `${i18n$('created_date')}: ${language.toDate(ujs.created_at)}`,
  4505. dataset: {
  4506. name: 'created_at',
  4507. value: new Date(ujs.created_at).toISOString()
  4508. }
  4509. });
  4510. const flicense = make('mu-js', 'mujs-list', {
  4511. title: ujs.license ?? i18n$('no_license'),
  4512. textContent: `${i18n$('license')}: ${ujs.license ?? i18n$('no_license')}`,
  4513. dataset: {
  4514. name: 'license'
  4515. }
  4516. });
  4517. const ftotal = make('mu-js', 'mujs-list', {
  4518. textContent: `${i18n$('total_installs')}: ${language.toNumber(ujs.total_installs)}`,
  4519. dataset: {
  4520. name: 'total_installs'
  4521. }
  4522. });
  4523. const fratings = make('mu-js', 'mujs-list', {
  4524. title: i18n$('ratings'),
  4525. textContent: `${i18n$('ratings')}:`
  4526. });
  4527. const fgood = make('mu-js', 'mujs-list mujs-ratings', {
  4528. title: i18n$('good'),
  4529. textContent: ujs.good_ratings,
  4530. dataset: {
  4531. name: 'good_ratings',
  4532. el: 'good'
  4533. }
  4534. });
  4535. const fok = make('mu-js', 'mujs-list mujs-ratings', {
  4536. title: i18n$('ok'),
  4537. textContent: ujs.ok_ratings,
  4538. dataset: {
  4539. name: 'ok_ratings',
  4540. el: 'ok'
  4541. }
  4542. });
  4543. const fbad = make('mu-js', 'mujs-list mujs-ratings', {
  4544. title: i18n$('bad'),
  4545. textContent: ujs.bad_ratings,
  4546. dataset: {
  4547. name: 'bad_ratings',
  4548. el: 'bad'
  4549. }
  4550. });
  4551. const fdesc = make('mu-js', 'mujs-list mujs-pointer', {
  4552. title: ujs.description,
  4553. textContent: ujs.description,
  4554. dataset: {
  4555. command: 'list-description'
  4556. }
  4557. });
  4558. const scriptInstall = make('mu-jsbtn', 'install', {
  4559. innerHTML: `${iconSVG.load('install')} ${i18n$('install')}`,
  4560. title: `${i18n$('install')} "${ujs.name}"`,
  4561. dataset: {
  4562. command: 'install-script',
  4563. userjs: ujs.id
  4564. }
  4565. });
  4566. const scriptDownload = make('mu-jsbtn', {
  4567. innerHTML: `${iconSVG.load('download')} ${i18n$('saveFile')}`,
  4568. dataset: {
  4569. command: 'download-userjs',
  4570. userjs: ujs.id,
  4571. userjsName: ujs.name
  4572. }
  4573. });
  4574. const tr = make('tr', 'frame', {
  4575. dataset: {
  4576. scriptId: ujs.id
  4577. }
  4578. });
  4579. const codeArea = make('textarea', 'code-area hidden', {
  4580. dataset: {
  4581. name: 'code'
  4582. },
  4583. rows: '10',
  4584. autocomplete: false,
  4585. spellcheck: false,
  4586. wrap: 'soft'
  4587. });
  4588. const loadCode = make('mu-jsbtn', {
  4589. innerHTML: `${iconSVG.load('code')} ${i18n$('code')}`,
  4590. dataset: {
  4591. command: 'load-userjs',
  4592. userjs: ujs.id
  4593. }
  4594. });
  4595. const loadMetadata = make('mu-jsbtn', {
  4596. innerHTML: `${iconSVG.load('code')} ${i18n$('metadata')}`,
  4597. dataset: {
  4598. command: 'load-header',
  4599. userjs: ujs.id
  4600. }
  4601. });
  4602. tr.dataset.engine = engine;
  4603. if (!engine.includes('fork') && cfg.recommend.others && goodUserJS.includes(ujs.url)) {
  4604. tr.dataset.good = 'upsell';
  4605. }
  4606. for (const u of ujs.users) {
  4607. const user = make('mujs-a', {
  4608. innerHTML: u.name,
  4609. title: u.url,
  4610. dataset: {
  4611. command: 'open-tab',
  4612. webpage: u.url
  4613. }
  4614. });
  4615. if (
  4616. cfg.recommend.author &&
  4617. (u.id === authorID || u.url === 'https://github.com/magicoflolis')
  4618. ) {
  4619. tr.dataset.author = 'upsell';
  4620. dom.prop(user, 'innerHTML', `${u.name} ${iconSVG.load('verified')}`);
  4621. }
  4622. uframe.append(user);
  4623. }
  4624. if (cfg.recommend.others && goodUserJS.includes(ujs.id)) {
  4625. tr.dataset.good = 'upsell';
  4626. }
  4627. eframe.append(scriptInstall);
  4628. ratings.append(fratings, fgood, fok, fbad);
  4629. jsInfo.append(ftotal, ratings, fver, fcreated);
  4630. mkList(i18n$('code_size'), {
  4631. list: ujs._mujs.code.code_size,
  4632. type: 'code_size',
  4633. root: jsInfo
  4634. });
  4635. jsInfoB.append(flicense);
  4636. const data_meta = ujs._mujs.code?.data_meta ?? {};
  4637. mkList(i18n$('antifeatures'), {
  4638. list: data_meta.antifeatures ?? [],
  4639. type: 'antifeatures',
  4640. root: jsInfoB
  4641. });
  4642. mkList(i18n$('applies_to'), {
  4643. list: ujs._mujs.code?.data_names ?? [],
  4644. type: 'data_names',
  4645. root: jsInfoB
  4646. });
  4647. mkList('@grant', {
  4648. list: data_meta.grant ?? [],
  4649. type: 'grant',
  4650. root: jsInfoB
  4651. });
  4652. mkList('@require', {
  4653. list: data_meta.require,
  4654. type: 'require',
  4655. root: jsInfoB
  4656. });
  4657. mkList('@resource', {
  4658. list: isNull(data_meta.resource) ? [] : [data_meta.resource],
  4659. type: 'resource',
  4660. root: jsInfoB
  4661. });
  4662. fmore.append(jsInfo, jsInfoB);
  4663. fBtns.append(scriptDownload, loadCode, loadMetadata);
  4664. fname.append(ftitle, fdesc, fmore, fBtns, codeArea);
  4665. fname._mujs = { fmore, fBtns, codeArea };
  4666. const loadPage = make('mu-jsbtn', {
  4667. innerHTML: `${iconSVG.load('pager')} Page`,
  4668. dataset: {
  4669. command: 'load-page',
  4670. userjs: ujs.id
  4671. }
  4672. });
  4673. fBtns.append(loadPage);
  4674. if (ujs._mujs.code?.translated) tr.classList.add('translated');
  4675. for (const e of [fname, uframe, fdaily, fupdated, eframe]) tr.append(e);
  4676. ujs._mujs.root = tr;
  4677. return ujs._mujs.root;
  4678. };
  4679. // #endregion
  4680. const loadFilters = () => {
  4681. /** @type {Map<string, import("../typings/types.d.ts").Filters >} */
  4682. const pool = new Map();
  4683. const handles = {
  4684. pool,
  4685. enabled() {
  4686. return [...pool.values()].filter((o) => o.enabled);
  4687. },
  4688. refresh() {
  4689. if (!Object.is(pool.size, 0)) pool.clear();
  4690. for (const [key, value] of Object.entries(cfg.filters)) {
  4691. if (!pool.has(key))
  4692. pool.set(key, {
  4693. ...value,
  4694. reg: new RegExp(value.regExp, value.flag),
  4695. keyReg: new RegExp(key.trim().toLocaleLowerCase(), 'gi'),
  4696. valueReg: new RegExp(value.name.trim().toLocaleLowerCase(), 'gi')
  4697. });
  4698. }
  4699. return this;
  4700. },
  4701. get(str) {
  4702. return [...pool.values()].find((v) => v.keyReg.test(str) || v.valueReg.test(str));
  4703. },
  4704. /**
  4705. * @param { import("../typings/types.d.ts").GSForkQuery } param0
  4706. */
  4707. match({ name, users }) {
  4708. const p = handles.enabled();
  4709. if (Object.is(p.length, 0)) return true;
  4710. for (const v of p) {
  4711. if ([{ name }, ...users].find((o) => o.name.match(v.reg))) return false;
  4712. }
  4713. return true;
  4714. }
  4715. };
  4716. for (const [key, value] of Object.entries(cfg.filters)) {
  4717. if (!pool.has(key))
  4718. pool.set(key, {
  4719. ...value,
  4720. reg: new RegExp(value.regExp, value.flag),
  4721. keyReg: new RegExp(key.trim().toLocaleLowerCase(), 'gi'),
  4722. valueReg: new RegExp(value.name.trim().toLocaleLowerCase(), 'gi')
  4723. });
  4724. }
  4725. return handles.refresh();
  4726. };
  4727. // #region List
  4728. class List {
  4729. intEngines;
  4730. intHost;
  4731. constructor(hostname = undefined) {
  4732. this.build = this.build.bind(this);
  4733. this.groupBy = this.groupBy.bind(this);
  4734. this.dispatch = this.dispatch.bind(this);
  4735. this.sortRecords = this.sortRecords.bind(this);
  4736. this.container = container;
  4737. this.intEngines = cfg.engines ?? [];
  4738. this.host = hostname;
  4739. }
  4740. setEngines(engines = []) {
  4741. const { host } = this;
  4742. return engines.filter((e) => {
  4743. if (!e.enabled) {
  4744. return false;
  4745. }
  4746. const v = engineUnsupported[e.name] ?? [];
  4747. if (v.includes(host)) {
  4748. showError(`Engine: "${e.name}" unsupported on "${host}"`);
  4749. container.timeoutFrame();
  4750. return false;
  4751. }
  4752. return true;
  4753. });
  4754. }
  4755. dispatch(ujs) {
  4756. const { CustomEvent } = _self;
  4757. const customEvent = new CustomEvent('updateditem', { detail: ujs });
  4758. main.dispatchEvent(customEvent);
  4759. }
  4760. set engines(engines) {
  4761. this.intEngines = this.setEngines(engines);
  4762. }
  4763. get engines() {
  4764. return this.intEngines;
  4765. }
  4766. set host(hostname) {
  4767. if (isEmpty(hostname)) hostname = this.container.host;
  4768. this.intHost = hostname;
  4769. this.blacklisted = this.container.checkBlacklist(hostname);
  4770. this.intEngines = this.setEngines(this.engines);
  4771. this.domain = this.getDomain(this.intHost);
  4772. if (this.blacklisted) {
  4773. showError(`Blacklisted "${hostname}"`);
  4774. this.container.timeoutFrame();
  4775. }
  4776. }
  4777. get host() {
  4778. return this.intHost;
  4779. }
  4780. /**
  4781. * @template { string } S
  4782. * @param { S } str
  4783. */
  4784. getDomain(str = '') {
  4785. if (str === '*') {
  4786. return 'all-sites';
  4787. }
  4788. return str.split('.').at(-2) ?? BLANK_PAGE;
  4789. }
  4790. // #region Builder
  4791. build() {
  4792. try {
  4793. container.refresh();
  4794. const { blacklisted, engines, host, domain, dispatch } = this;
  4795. if (blacklisted || isEmpty(engines)) {
  4796. container.opacityMin = '0';
  4797. mainframe.style.opacity = container.opacityMin;
  4798. return;
  4799. }
  4800. const fetchRecords = [];
  4801. const bsFilter = loadFilters();
  4802. const hostCache = Array.from(this);
  4803. info('Building list', { hostCache, engines, container, list: this });
  4804. const g = this.groupBy();
  4805. const toFetch = engines.filter((engine) => !g[engine.name]);
  4806. const isCached = toFetch.filter((engine) =>
  4807. hostCache.find(({ _mujs }) => engine.name === _mujs.info.engine.name)
  4808. );
  4809. if (!isBlank(toFetch) && isBlank(isCached)) {
  4810. for (const engine of engines) {
  4811. info(`Fetching from "${engine.name}" for "${host}"`);
  4812. const respError = (error) => {
  4813. if (!error.cause) error.cause = engine.name;
  4814. if (error.message.startsWith('429')) {
  4815. showError(`Engine: "${engine.name}" Too many requests...`);
  4816. return;
  4817. }
  4818. showError(`Engine: "${engine.name}"`, error.message);
  4819. };
  4820. const _mujs = (d) => {
  4821. /**
  4822. * @type {import("../typings/types.d.ts").GSForkQuery}
  4823. */
  4824. const ujs = {
  4825. ...template,
  4826. ...d,
  4827. code_urls: [],
  4828. _mujs: {
  4829. root: {},
  4830. info: {
  4831. engine,
  4832. host
  4833. },
  4834. code: {
  4835. meta: {}
  4836. }
  4837. }
  4838. };
  4839. ujs._mujs.code.request = async (translate = false, code_url) => {
  4840. if (typeof ujs._mujs.code.data_code_block === 'string') {
  4841. return ujs._mujs.code;
  4842. }
  4843. const p = new ParseUserJS();
  4844. await p.request(translate, code_url ?? ujs.code_url, ujs);
  4845. if (code_url) {
  4846. return p;
  4847. }
  4848. for (const [k, v] of Object.entries(p)) ujs._mujs.code[k] = v;
  4849. return ujs._mujs.code;
  4850. };
  4851. return ujs;
  4852. };
  4853. /**
  4854. * Prior to UserScript v7.0.0
  4855. * @template {string} F
  4856. * @param {F} fallback
  4857. * @returns {F}
  4858. */
  4859. const toQuery = (fallback) => {
  4860. if (engine.query) {
  4861. return decode(engine.query)
  4862. .replace(/\{host\}/g, host)
  4863. .replace(/\{domain\}/g, domain);
  4864. }
  4865. return fallback;
  4866. };
  4867. /**
  4868. * @param { import("../typings/types.d.ts").GSFork } dataQ
  4869. */
  4870. const forkFN = async (dataQ) => {
  4871. if (!dataQ) {
  4872. showError('Invalid data received from the server, check internet connection');
  4873. return;
  4874. }
  4875. const dq = Array.isArray(dataQ)
  4876. ? dataQ
  4877. : Array.isArray(dataQ.query)
  4878. ? dataQ.query
  4879. : [];
  4880. const dataA = dq
  4881. .filter(Boolean)
  4882. .filter((d) => !d.deleted)
  4883. .filter(bsFilter.match);
  4884. if (isBlank(dataA)) {
  4885. return;
  4886. }
  4887. const { groupBy } = _self;
  4888. /**
  4889. * @type { {[key: string]: import("../typings/types.d.ts").GSForkQuery[]} }
  4890. */
  4891. const g = groupBy(dataA.map(_mujs), ({ locale }) => {
  4892. return locale.split('-')[0] ?? locale;
  4893. });
  4894. for (const [k, list] of Object.entries(g)) {
  4895. for (const ujs of list) {
  4896. if (cfg.filterlang && k !== language.current) {
  4897. const c = await ujs._mujs.code.request(true);
  4898. if (!c.translated) continue;
  4899. }
  4900. if (
  4901. !ujs._mujs.code.data_code_block &&
  4902. (cfg.preview.code || cfg.preview.metadata)
  4903. ) {
  4904. ujs._mujs.code.request().then(() => {
  4905. dispatch(ujs);
  4906. });
  4907. }
  4908. if (isUserCSS(ujs.code_url)) {
  4909. ujs.code_urls.push(
  4910. {
  4911. name: `${ujs.name} (.user.css)`,
  4912. code_url: ujs.code_url
  4913. },
  4914. {
  4915. name: `${ujs.name} (.user.js)`,
  4916. code_url: ujs.code_url.replace(/\.user\.css$/, '.user.js')
  4917. }
  4918. );
  4919. }
  4920. createjs(ujs, engine.name);
  4921. }
  4922. }
  4923. };
  4924. /**
  4925. * @param {Document} htmlDocument
  4926. */
  4927. const openuserjs = async (htmlDocument) => {
  4928. try {
  4929. if (!htmlDocument) {
  4930. showError('Invalid data received from the server, TODO fix this');
  4931. return;
  4932. }
  4933. const selected = htmlDocument.documentElement;
  4934. if (/openuserjs/gi.test(engine.name)) {
  4935. const col = qsA('.col-sm-8 .tr-link', selected) ?? [];
  4936. for (const i of col) {
  4937. while (isNull(qs('.script-version', i))) {
  4938. await new Promise((resolve) => requestAnimationFrame(resolve));
  4939. }
  4940. const fixurl = dom
  4941. .prop(qs('.tr-link-a', i), 'href')
  4942. .replace(
  4943. new RegExp(document.location.origin, 'gi'),
  4944. 'https://openuserjs.org'
  4945. );
  4946. const ujs = _mujs({
  4947. name: dom.text(qs('.tr-link-a', i)),
  4948. description: dom.text(qs('p', i)),
  4949. version: dom.text(qs('.script-version', i)),
  4950. url: fixurl,
  4951. code_url: `${fixurl.replace(/\/scripts/gi, '/install')}.user.js`,
  4952. total_installs: dom.text(qs('td:nth-child(2) p', i)),
  4953. created_at: dom.attr(qs('td:nth-child(4) time', i), 'datetime'),
  4954. code_updated_at: dom.attr(qs('td:nth-child(4) time', i), 'datetime'),
  4955. users: [
  4956. {
  4957. name: dom.text(qs('.inline-block a', i)),
  4958. url: dom.prop(qs('.inline-block a', i), 'href')
  4959. }
  4960. ]
  4961. });
  4962. if (bsFilter.match(ujs)) {
  4963. continue;
  4964. }
  4965. if (
  4966. !ujs._mujs.code.data_code_block &&
  4967. (cfg.preview.code || cfg.preview.metadata)
  4968. ) {
  4969. ujs._mujs.code.request().then(() => {
  4970. dispatch(ujs);
  4971. });
  4972. }
  4973. createjs(ujs, engine.name);
  4974. }
  4975. }
  4976. } catch (ex) {
  4977. showError(ex);
  4978. }
  4979. };
  4980. const gitFN = (data) => {
  4981. try {
  4982. if (isBlank(data.items)) {
  4983. return;
  4984. }
  4985. for (const r of data.items) {
  4986. const ujs = _mujs({
  4987. id: r.id ?? 0,
  4988. name: r.name,
  4989. description: isEmpty(r.description) ? i18n$('no_license') : r.description,
  4990. url: r.html_url,
  4991. code_url: r.html_url,
  4992. page_url: `${r.url}/contents/README.md`,
  4993. created_at: r.created_at,
  4994. code_updated_at: r.updated_at || Date.now(),
  4995. daily_installs: r.watchers_count ?? 0,
  4996. good_ratings: r.stargazers_count ?? 0,
  4997. users: [
  4998. {
  4999. name: r.owner.login,
  5000. url: r.owner.html_url
  5001. }
  5002. ]
  5003. });
  5004. if (r.license?.name) ujs.license = r.license.name;
  5005. const rootPath = r.contents_url.replace(/\{\+path\}/, '');
  5006. const fetchContent = async (dir) => {
  5007. const contents = await Network.req(dir, 'GET', 'json', {
  5008. headers: {
  5009. Accept: 'application/vnd.github+json',
  5010. Authorization: `Bearer ${engine.token}`,
  5011. 'X-GitHub-Api-Version': '2022-11-28'
  5012. }
  5013. }).catch(respError);
  5014. for (const content of contents) {
  5015. if (content.type === 'file') {
  5016. if (isUserJS(content.name)) {
  5017. ujs.code_urls.push({
  5018. name: content.name,
  5019. code_url: content.download_url
  5020. });
  5021. } else if (isUserCSS(content.name)) {
  5022. ujs.code_urls.push({
  5023. name: content.name,
  5024. code_url: content.download_url
  5025. });
  5026. }
  5027. } else if (content.type === 'dir') {
  5028. await fetchContent(`${rootPath}/${content.path}`);
  5029. }
  5030. }
  5031. };
  5032. fetchContent(rootPath).then(() => {
  5033. if (isEmpty(ujs.code_urls)) {
  5034. ujs.deleted = true;
  5035. } else if (
  5036. !ujs._mujs.code.data_code_block &&
  5037. (cfg.preview.code || cfg.preview.metadata)
  5038. ) {
  5039. ujs._mujs.code.request().then(() => {
  5040. dispatch(ujs);
  5041. });
  5042. return;
  5043. }
  5044. dispatch(ujs);
  5045. });
  5046. createjs(ujs, engine.name);
  5047. }
  5048. } catch (ex) {
  5049. showError(ex);
  5050. }
  5051. };
  5052. let netFN;
  5053. if (/github/gi.test(engine.name)) {
  5054. if (isEmpty(engine.token)) {
  5055. showError(`"${engine.name}" requires a token to use`);
  5056. continue;
  5057. }
  5058. Network.req(
  5059. `https://api.github.com/search/repositories?q=topic:${domain}+topic:userstyle`,
  5060. 'GET',
  5061. 'json',
  5062. {
  5063. headers: {
  5064. Accept: 'application/vnd.github+json',
  5065. Authorization: `Bearer ${engine.token}`,
  5066. 'X-GitHub-Api-Version': '2022-11-28'
  5067. }
  5068. }
  5069. )
  5070. .then(gitFN)
  5071. .catch(respError);
  5072. netFN = Network.req(
  5073. toQuery(
  5074. `https://api.github.com/search/repositories?q=topic:${domain}+topic:userscript`
  5075. ),
  5076. 'GET',
  5077. 'json',
  5078. {
  5079. headers: {
  5080. Accept: 'application/vnd.github+json',
  5081. Authorization: `Bearer ${engine.token}`,
  5082. 'X-GitHub-Api-Version': '2022-11-28'
  5083. }
  5084. }
  5085. )
  5086. .then(gitFN)
  5087. .then(() => {
  5088. Network.req('https://api.github.com/rate_limit', 'GET', 'json', {
  5089. headers: {
  5090. Accept: 'application/vnd.github+json',
  5091. Authorization: `Bearer ${engine.token}`,
  5092. 'X-GitHub-Api-Version': '2022-11-28'
  5093. }
  5094. })
  5095. .then((data) => {
  5096. for (const [key, value] of Object.entries(data.resources.code_search)) {
  5097. const txt = make('mujs-row', 'rate-info', {
  5098. textContent: `${key.toUpperCase()}: ${value}`
  5099. });
  5100. rateContainer.append(txt);
  5101. }
  5102. })
  5103. .catch(respError);
  5104. });
  5105. } else if (/openuserjs/gi.test(engine.name)) {
  5106. netFN = Network.req(toQuery(`${engine.url}${host}`), 'GET', 'document').then(
  5107. openuserjs
  5108. );
  5109. } else {
  5110. netFN = Network.req(
  5111. toQuery(`${engine.url}/scripts/by-site/${host}.json?language=all`)
  5112. ).then(forkFN);
  5113. }
  5114. if (netFN) {
  5115. fetchRecords.push(netFN.catch(respError));
  5116. }
  5117. }
  5118. } else {
  5119. for (const ujs of hostCache) tabbody.append(ujs._mujs.root);
  5120. }
  5121. urlBar.placeholder = i18n$('search_placeholder');
  5122. urlBar.value = '';
  5123. if (isBlank(fetchRecords)) {
  5124. this.sortRecords();
  5125. return;
  5126. }
  5127. Promise.allSettled(fetchRecords).then(this.sortRecords).catch(showError);
  5128. } catch (ex) {
  5129. showError(ex);
  5130. }
  5131. }
  5132. // #endregion
  5133. sortRecords() {
  5134. const arr = Array.from(this);
  5135. for (const ujs of arr.flat().sort((a, b) => {
  5136. const sortType = cfg.autoSort ?? 'daily_installs';
  5137. return b[sortType] - a[sortType];
  5138. })) {
  5139. if (isElem(ujs._mujs.root)) tabbody.append(ujs._mujs.root);
  5140. }
  5141. for (const [name, value] of Object.entries(this.groupBy(arr)))
  5142. Counter.update(value.length, { name });
  5143. }
  5144. groupBy() {
  5145. const { groupBy } = _self;
  5146. return groupBy(Array.from(this), ({ _mujs }) => _mujs.info.engine.name);
  5147. }
  5148. *[Symbol.iterator]() {
  5149. const { intHost, engines } = this;
  5150. const arr = Array.from(this.container).filter(
  5151. ({ _mujs }) =>
  5152. _mujs.info.host === intHost &&
  5153. engines.find((engine) => engine.enabled && engine.name === _mujs.info.engine.name)
  5154. );
  5155. for (const userjs of arr) {
  5156. yield userjs;
  5157. }
  5158. }
  5159. }
  5160. const MUList = new List();
  5161. // #endregion
  5162. // #region Make Config
  5163. const makecfg = () => {
  5164. const cbtn = make('mu-js', 'mujs-sty-flex');
  5165. const savebtn = make('mujs-btn', 'save', {
  5166. textContent: i18n$('save'),
  5167. dataset: {
  5168. command: 'save'
  5169. },
  5170. disabled: false
  5171. });
  5172. const resetbtn = make('mujs-btn', 'reset', {
  5173. textContent: i18n$('reset'),
  5174. dataset: {
  5175. command: 'reset'
  5176. }
  5177. });
  5178. cbtn.append(resetbtn, savebtn);
  5179. const makesection = (name, tag) => {
  5180. tag = tag ?? i18n$('no_license');
  5181. name = name ?? i18n$('no_license');
  5182. const sec = make('mujs-section', {
  5183. dataset: {
  5184. name: tag
  5185. }
  5186. });
  5187. const lb = make('label', {
  5188. dataset: {
  5189. command: tag
  5190. }
  5191. });
  5192. const divDesc = make('mu-js', {
  5193. textContent: name
  5194. });
  5195. ael(sec, 'click', (evt) => {
  5196. /** @type { HTMLElement } */
  5197. const target = evt.target.closest('[data-command]');
  5198. if (!target) {
  5199. return;
  5200. }
  5201. const cmd = target.dataset.command;
  5202. if (cmd === tag) {
  5203. const a = qsA(`[data-${tag}]`, sec);
  5204. if (dom.cl.has(a, 'hidden')) {
  5205. dom.cl.remove(a, 'hidden');
  5206. } else {
  5207. dom.cl.add(a, 'hidden');
  5208. }
  5209. }
  5210. });
  5211. lb.append(divDesc);
  5212. sec.append(lb);
  5213. cfgpage.append(sec);
  5214. !cfgSec.has(sec) && cfgSec.add(sec);
  5215. return sec;
  5216. };
  5217. const sections = {
  5218. general: makesection('General', 'general'),
  5219. load: makesection('Automation', 'load'),
  5220. list: makesection('List', 'list'),
  5221. filters: makesection('List Filters', 'filters'),
  5222. blacklist: makesection('Blacklist (WIP)', 'blacklist'),
  5223. engine: makesection('Search Engines', 'engine'),
  5224. theme: makesection('Theme Colors', 'theme'),
  5225. exp: makesection('Import / Export', 'exp')
  5226. };
  5227. const makeRow = (text, value, type = 'checkbox', tag = 'general', attrs = {}) => {
  5228. const lb = make('label', 'sub-section hidden', {
  5229. textContent: text,
  5230. dataset: {
  5231. [tag]: text
  5232. }
  5233. });
  5234. const getDefault = () => {
  5235. if (tag === 'engine') {
  5236. const engine = DEFAULT_CONFIG.engines.find((engine) => engine.name === value);
  5237. if (engine) {
  5238. return engine;
  5239. }
  5240. }
  5241. const nm = /^(\w+)-(.+)/.exec(value);
  5242. if (nm) {
  5243. return DEFAULT_CONFIG[nm[1]][nm[2]];
  5244. }
  5245. return DEFAULT_CONFIG[value];
  5246. };
  5247. const getValue = () => {
  5248. if (tag === 'engine') {
  5249. const engine = cfg.engines.find((engine) => engine.name === value);
  5250. if (engine) {
  5251. return engine;
  5252. }
  5253. }
  5254. const nm = /^(\w+)-(.+)/.exec(value);
  5255. if (nm) {
  5256. return cfg[nm[1]][nm[2]];
  5257. }
  5258. return cfg[value];
  5259. };
  5260. const obj = {
  5261. text,
  5262. tag,
  5263. value,
  5264. type,
  5265. attrs,
  5266. default: getDefault(),
  5267. cache: getValue()
  5268. };
  5269. if (type === 'select') {
  5270. const inp = make('select', {
  5271. dataset: {
  5272. [tag]: text
  5273. },
  5274. ...attrs
  5275. });
  5276. for (const selV of Object.keys(template)) {
  5277. if (selV === 'deleted' || selV === 'users') continue;
  5278. const o = make('option', {
  5279. value: selV,
  5280. textContent: selV
  5281. });
  5282. inp.append(o);
  5283. }
  5284. inp.value = cfg[value];
  5285. lb.append(inp);
  5286. if (sections[tag]) {
  5287. sections[tag].append(lb);
  5288. }
  5289. obj.elem = inp;
  5290. cfgBase.push(obj);
  5291. return lb;
  5292. }
  5293. const inp = make('input', {
  5294. type,
  5295. dataset: {
  5296. [tag]: text
  5297. },
  5298. ...attrs
  5299. });
  5300. if (tag === 'engine') {
  5301. inp.dataset.name = value;
  5302. }
  5303. if (sections[tag]) {
  5304. sections[tag].append(lb);
  5305. }
  5306. if (type === 'checkbox') {
  5307. const inlab = make('mu-js', 'mujs-inlab');
  5308. const la = make('label', {
  5309. onclick() {
  5310. inp.dispatchEvent(new MouseEvent('click'));
  5311. }
  5312. });
  5313. inlab.append(inp, la);
  5314. lb.append(inlab);
  5315. const nm = /^(\w+)-(.+)/.exec(value);
  5316. if (nm) {
  5317. if (nm[1] === 'filters') {
  5318. inp.checked = cfg[nm[1]][nm[2]].enabled;
  5319. } else {
  5320. inp.checked = cfg[nm[1]][nm[2]];
  5321. }
  5322. } else {
  5323. inp.checked = cfg[value];
  5324. }
  5325. ael(inp, 'change', (evt) => {
  5326. container.unsaved = true;
  5327. if (/filterlang/i.test(value)) {
  5328. container.rebuild = true;
  5329. }
  5330. if (nm) {
  5331. if (nm[1] === 'filters') {
  5332. cfg[nm[1]][nm[2]].enabled = evt.target.checked;
  5333. } else {
  5334. cfg[nm[1]][nm[2]] = evt.target.checked;
  5335. }
  5336. } else {
  5337. cfg[value] = evt.target.checked;
  5338. }
  5339. });
  5340. if (tag === 'engine') {
  5341. const engine = cfg.engines.find((engine) => engine.name === value);
  5342. if (engine) {
  5343. inp.checked = engine.enabled;
  5344. inp.dataset.engine = engine.name;
  5345. ael(inp, 'change', (evt) => {
  5346. container.unsaved = true;
  5347. container.rebuild = true;
  5348. engine.enabled = evt.target.checked;
  5349. MUList.engines = cfg.engines;
  5350. });
  5351. if (engine.query) {
  5352. const d = DEFAULT_CONFIG.engines.find((e) => e.name === engine.name);
  5353. const urlInp = make('input', {
  5354. type: 'text',
  5355. defaultValue: '',
  5356. value: decode(engine.query),
  5357. placeholder: decode(d.query),
  5358. dataset: {
  5359. name: nm,
  5360. engine: engine.name
  5361. },
  5362. onchange(evt) {
  5363. container.unsaved = true;
  5364. container.rebuild = true;
  5365. try {
  5366. engine.query = encodeURIComponent(new URL(evt.target.value).toString());
  5367. MUList.engines = cfg.engines;
  5368. } catch (ex) {
  5369. err(ex);
  5370. }
  5371. }
  5372. });
  5373. obj.elemUrl = urlInp;
  5374. lb.append(urlInp);
  5375. }
  5376. if (engine.name === 'github') {
  5377. const ghToken = make('input', {
  5378. type: 'text',
  5379. defaultValue: '',
  5380. value: engine.token ?? '',
  5381. placeholder: 'Paste Access Token',
  5382. dataset: {
  5383. engine: 'github-token'
  5384. },
  5385. onchange(evt) {
  5386. container.unsaved = true;
  5387. container.rebuild = true;
  5388. engine.token = evt.target.value;
  5389. MUList.engines = cfg.engines;
  5390. }
  5391. });
  5392. obj.elemToken = ghToken;
  5393. lb.append(ghToken);
  5394. }
  5395. }
  5396. }
  5397. } else {
  5398. if (type === 'text') {
  5399. inp.defaultValue = '';
  5400. inp.value = value ?? '';
  5401. inp.placeholder = value ?? '';
  5402. if (tag === 'theme') {
  5403. inp.dataset[tag] = text;
  5404. ael(inp, 'change', (evt) => {
  5405. let isvalid = true;
  5406. try {
  5407. const val = evt.target.value;
  5408. const sty = container.root.style;
  5409. const str = `--mujs-${text}`;
  5410. const prop = sty.getPropertyValue(str);
  5411. if (isEmpty(val)) {
  5412. cfg.theme[text] = DEFAULT_CONFIG.theme[text];
  5413. sty.removeProperty(str);
  5414. return;
  5415. }
  5416. if (prop === val) {
  5417. return;
  5418. }
  5419. sty.removeProperty(str);
  5420. sty.setProperty(str, val);
  5421. cfg.theme[text] = val;
  5422. } catch (ex) {
  5423. err(ex);
  5424. isvalid = false;
  5425. } finally {
  5426. if (isvalid) {
  5427. dom.cl.remove(evt.target, 'mujs-invalid');
  5428. dom.prop(savebtn, 'disabled', false);
  5429. } else {
  5430. dom.cl.add(evt.target, 'mujs-invalid');
  5431. dom.prop(savebtn, 'disabled', true);
  5432. }
  5433. }
  5434. });
  5435. }
  5436. }
  5437. lb.append(inp);
  5438. }
  5439. obj.elem = inp;
  5440. cfgBase.push(obj);
  5441. return lb;
  5442. };
  5443. if (isGM) {
  5444. makeRow(i18n$('userjs_sync'), 'cache');
  5445. makeRow(i18n$('userjs_autoinject'), 'autoinject', 'checkbox', 'load');
  5446. }
  5447. makeRow(i18n$('redirect'), 'sleazyredirect');
  5448. makeRow(`${i18n$('dtime')} (ms)`, 'time', 'number', 'general', {
  5449. defaultValue: 10000,
  5450. value: cfg.time,
  5451. min: 0,
  5452. step: 500,
  5453. onbeforeinput(evt) {
  5454. if (evt.target.validity.badInput) {
  5455. dom.cl.add(evt.target, 'mujs-invalid');
  5456. dom.prop(savebtn, 'disabled', true);
  5457. } else {
  5458. dom.cl.remove(evt.target, 'mujs-invalid');
  5459. dom.prop(savebtn, 'disabled', false);
  5460. }
  5461. },
  5462. oninput(evt) {
  5463. container.unsaved = true;
  5464. const t = evt.target;
  5465. if (t.validity.badInput || (t.validity.rangeUnderflow && t.value !== '-1')) {
  5466. dom.cl.add(t, 'mujs-invalid');
  5467. dom.prop(savebtn, 'disabled', true);
  5468. } else {
  5469. dom.cl.remove(t, 'mujs-invalid');
  5470. dom.prop(savebtn, 'disabled', false);
  5471. cfg.time = isEmpty(t.value) ? cfg.time : parseFloat(t.value);
  5472. }
  5473. }
  5474. });
  5475. makeRow(i18n$('auto_fetch'), 'autofetch', 'checkbox', 'load');
  5476. makeRow(i18n$('userjs_fullscreen'), 'autoexpand', 'checkbox', 'load', {
  5477. onchange(e) {
  5478. if (e.target.checked) {
  5479. dom.cl.add([btnfullscreen, main], 'expanded');
  5480. dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('collapse'));
  5481. } else {
  5482. dom.cl.remove([btnfullscreen, main], 'expanded');
  5483. dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('expand'));
  5484. }
  5485. }
  5486. });
  5487. makeRow('Clear on Tab close', 'clearTabCache', 'checkbox', 'load');
  5488. makeRow(i18n$('default_sort'), 'autoSort', 'select', 'list');
  5489. makeRow(i18n$('filter'), 'filterlang', 'checkbox', 'list');
  5490. makeRow(i18n$('preview_code'), 'preview-code', 'checkbox', 'list');
  5491. makeRow(i18n$('preview_metadata'), 'preview-metadata', 'checkbox', 'list');
  5492. makeRow(i18n$('recommend_author'), 'recommend-author', 'checkbox', 'list');
  5493. makeRow(i18n$('recommend_other'), 'recommend-others', 'checkbox', 'list');
  5494. for (const [k, v] of Object.entries(cfg.filters))
  5495. makeRow(v.name, `filters-${k}`, 'checkbox', 'filters');
  5496. makeRow('Greasy Fork', 'greasyfork', 'checkbox', 'engine');
  5497. makeRow('Sleazy Fork', 'sleazyfork', 'checkbox', 'engine');
  5498. makeRow('Open UserJS', 'openuserjs', 'checkbox', 'engine');
  5499. makeRow('GitHub API', 'github', 'checkbox', 'engine');
  5500. for (const [k, v] of Object.entries(cfg.theme)) makeRow(k, v, 'text', 'theme');
  5501. // const blacklist = make('textarea', {
  5502. // dataset: {
  5503. // name: 'blacklist'
  5504. // },
  5505. // rows: '10',
  5506. // autocomplete: false,
  5507. // spellcheck: false,
  5508. // wrap: 'soft',
  5509. // value: JSON.stringify(cfg.blacklist, null, ' '),
  5510. // oninput(evt) {
  5511. // let isvalid = true;
  5512. // try {
  5513. // cfg.blacklist = JSON.parse(evt.target.value);
  5514. // isvalid = true;
  5515. // } catch (ex) {
  5516. // err(ex);
  5517. // isvalid = false;
  5518. // } finally {
  5519. // if (isvalid) {
  5520. // dom.cl.remove(evt.target, 'mujs-invalid');
  5521. // dom.prop(savebtn, 'disabled', false);
  5522. // } else {
  5523. // dom.cl.add(evt.target, 'mujs-invalid');
  5524. // dom.prop(savebtn, 'disabled', true);
  5525. // }
  5526. // }
  5527. // }
  5528. // });
  5529. // const addList = make('mujs-add', {
  5530. // textContent: '+',
  5531. // dataset: {
  5532. // command: 'new-list'
  5533. // }
  5534. // });
  5535. // const n = make('input', {
  5536. // type: 'text',
  5537. // defaultValue: '',
  5538. // value: '',
  5539. // placeholder: 'Name',
  5540. // });
  5541. // const inpValue = make('input', {
  5542. // type: 'text',
  5543. // defaultValue: '',
  5544. // value: '',
  5545. // placeholder: 'Value',
  5546. // });
  5547. // const label = make('label', 'new-list hidden', {
  5548. // dataset: {
  5549. // blacklist: 'new-list'
  5550. // }
  5551. // });
  5552. // label.append(n, inpValue, addList);
  5553. // listSec.append(label);
  5554. // ael(addList, 'click', () => {
  5555. // if (isEmpty(n.value) || isEmpty(inpValue.value)) {
  5556. // return
  5557. // };
  5558. // createList(n.value, n.value, inpValue.value);
  5559. // });
  5560. const createList = (key, v = '', disabled = false, type = 'String') => {
  5561. let txt = key;
  5562. if (typeof key === 'string') {
  5563. if (key.startsWith('userjs-')) {
  5564. disabled = true;
  5565. const s = key.substring(7);
  5566. txt = `Built-in "${s}"`;
  5567. v = builtinList[s];
  5568. }
  5569. } else {
  5570. if (!key.enabled) {
  5571. return;
  5572. }
  5573. }
  5574. if (isRegExp(v)) {
  5575. v = v.toString();
  5576. type = 'RegExp';
  5577. } else {
  5578. v = JSON.stringify(v);
  5579. type = 'Object';
  5580. }
  5581. const lb = make('label', 'hidden', {
  5582. textContent: txt,
  5583. dataset: {
  5584. blacklist: key
  5585. }
  5586. });
  5587. const inp = make('input', {
  5588. type: 'text',
  5589. defaultValue: '',
  5590. value: v ?? '',
  5591. placeholder: v ?? '',
  5592. dataset: {
  5593. blacklist: key
  5594. },
  5595. onchange(evt) {
  5596. let isvalid = true;
  5597. try {
  5598. const val = evt.target.value;
  5599. if (isEmpty(val)) {
  5600. return;
  5601. }
  5602. isvalid = true;
  5603. } catch (ex) {
  5604. err(ex);
  5605. isvalid = false;
  5606. } finally {
  5607. if (isvalid) {
  5608. dom.cl.remove(evt.target, 'mujs-invalid');
  5609. dom.prop(savebtn, 'disabled', false);
  5610. } else {
  5611. dom.cl.add(evt.target, 'mujs-invalid');
  5612. dom.prop(savebtn, 'disabled', true);
  5613. }
  5614. }
  5615. }
  5616. });
  5617. const selType = make('select', {
  5618. disabled,
  5619. dataset: {
  5620. blacklist: key
  5621. }
  5622. });
  5623. if (disabled) {
  5624. inp.readOnly = true;
  5625. const o = make('option', {
  5626. value: type,
  5627. textContent: type
  5628. });
  5629. selType.append(o);
  5630. } else {
  5631. for (const selV of ['String', 'RegExp', 'Object']) {
  5632. const o = make('option', {
  5633. value: selV,
  5634. textContent: selV
  5635. });
  5636. selType.append(o);
  5637. }
  5638. }
  5639. selType.value = type;
  5640. lb.append(inp, selType);
  5641. sections.blacklist.append(lb);
  5642. };
  5643. for (const key of cfg.blacklist) createList(key);
  5644. const transfers = {
  5645. export: {
  5646. cfg: make('mujs-btn', 'mujs-export sub-section hidden', {
  5647. textContent: i18n$('export_config'),
  5648. dataset: {
  5649. command: 'export-cfg',
  5650. exp: 'export-cfg'
  5651. }
  5652. }),
  5653. theme: make('mujs-btn', 'mujs-export sub-section hidden', {
  5654. textContent: i18n$('export_theme'),
  5655. dataset: {
  5656. command: 'export-theme',
  5657. exp: 'export-theme'
  5658. }
  5659. })
  5660. },
  5661. import: {
  5662. cfg: make('mujs-btn', 'mujs-import sub-section hidden', {
  5663. textContent: i18n$('import_config'),
  5664. dataset: {
  5665. command: 'import-cfg',
  5666. exp: 'import-cfg'
  5667. }
  5668. }),
  5669. theme: make('mujs-btn', 'mujs-import sub-section hidden', {
  5670. textContent: i18n$('import_theme'),
  5671. dataset: {
  5672. command: 'import-theme',
  5673. exp: 'import-theme'
  5674. }
  5675. })
  5676. }
  5677. };
  5678. for (const value of Object.values(transfers)) {
  5679. for (const v of Object.values(value)) {
  5680. sections.exp.append(v);
  5681. }
  5682. }
  5683. cfgpage.append(cbtn);
  5684. };
  5685. // #endregion
  5686. container.Tabs.custom = (host) => {
  5687. MUList.host = host;
  5688. respHandles.build();
  5689. };
  5690. ael(mainframe, 'mouseenter', (evt) => {
  5691. evt.preventDefault();
  5692. evt.stopPropagation();
  5693. evt.target.style.opacity = container.opacityMax;
  5694. frameTimeout.clear(...frameTimeout.ids);
  5695. });
  5696. ael(mainframe, 'mouseleave', (evt) => {
  5697. evt.preventDefault();
  5698. evt.stopPropagation();
  5699. evt.target.style.opacity = container.opacityMin;
  5700. container.timeoutFrame();
  5701. });
  5702. ael(mainframe, 'click', (evt) => {
  5703. evt.preventDefault();
  5704. frameTimeout.clear(...frameTimeout.ids);
  5705. dom.cl.remove(main, 'hidden');
  5706. dom.cl.add(mainframe, 'hidden');
  5707. if (cfg.autoexpand) {
  5708. dom.cl.add([btnfullscreen, main], 'expanded');
  5709. dom.prop(btnfullscreen, 'innerHTML', iconSVG.load('collapse'));
  5710. }
  5711. });
  5712. ael(urlBar, 'input', (evt) => {
  5713. evt.preventDefault();
  5714. if (urlBar.placeholder === i18n$('newTab')) {
  5715. return;
  5716. }
  5717. /**
  5718. * @type { string }
  5719. */
  5720. const val = evt.target.value;
  5721. const section = qsA('mujs-section[data-name]', cfgpage);
  5722. if (isEmpty(val)) {
  5723. dom.cl.remove(container.toElem(), 'hidden');
  5724. dom.cl.remove(section, 'hidden');
  5725. return;
  5726. }
  5727. const finds = new Set();
  5728. if (!dom.cl.has(cfgpage, 'hidden')) {
  5729. const reg = new RegExp(val, 'gi');
  5730. for (const elem of section) {
  5731. if (!isElem(elem)) continue;
  5732. if (finds.has(elem)) continue;
  5733. if (elem.dataset.name.match(reg)) finds.add(elem);
  5734. }
  5735. dom.cl.add(section, 'hidden');
  5736. dom.cl.remove([...finds], 'hidden');
  5737. return;
  5738. }
  5739. const cacheValues = Array.from(container).filter(({ _mujs }) => {
  5740. return !finds.has(_mujs.root);
  5741. });
  5742. /**
  5743. * @param {RegExpMatchArray} regExp
  5744. * @param {keyof import("../typings/types.d.ts").GSForkQuery} key
  5745. */
  5746. const ezQuery = (regExp, key) => {
  5747. const q_value = val.replace(regExp, '');
  5748. const reg = new RegExp(q_value, 'gi');
  5749. for (const v of cacheValues) {
  5750. let k = v[key];
  5751. if (typeof k === 'number') {
  5752. k = `${v[key]}`;
  5753. }
  5754. if (k && k.match(reg)) {
  5755. finds.add(v._mujs.root);
  5756. }
  5757. }
  5758. };
  5759. if (val.match(/^(code_url|url):/)) {
  5760. ezQuery(/^(code_url|url):/, 'code_url');
  5761. } else if (val.match(/^(author|users?):/)) {
  5762. const parts = /^[\w_]+:(.+)/.exec(val);
  5763. if (parts) {
  5764. const reg = new RegExp(parts[1], 'gi');
  5765. for (const v of cacheValues.filter((v) => !isEmpty(v.users))) {
  5766. for (const user of v.users) {
  5767. for (const value of Object.values(user)) {
  5768. if (typeof value === 'string' && value.match(reg)) {
  5769. finds.add(v._mujs.root);
  5770. } else if (typeof value === 'number' && `${value}`.match(reg)) {
  5771. finds.add(v._mujs.root);
  5772. }
  5773. }
  5774. }
  5775. }
  5776. }
  5777. } else if (val.match(/^(locale|i18n):/)) {
  5778. ezQuery(/^(locale|i18n):/, 'locale');
  5779. } else if (val.match(/^id:/)) {
  5780. ezQuery(/^id:/, 'id');
  5781. } else if (val.match(/^license:/)) {
  5782. ezQuery(/^license:/, 'license');
  5783. } else if (val.match(/^name:/)) {
  5784. ezQuery(/^name:/, 'name');
  5785. } else if (val.match(/^description:/)) {
  5786. ezQuery(/^description:/, 'description');
  5787. } else if (val.match(/^(search_engine|engine):/)) {
  5788. const parts = /^[\w_]+:(\w+)/.exec(val);
  5789. if (parts) {
  5790. const reg = new RegExp(parts[1], 'gi');
  5791. for (const { _mujs } of cacheValues)
  5792. if (_mujs.info.engine.name.match(reg)) finds.add(_mujs.root);
  5793. }
  5794. } else if (val.match(/^filter:/)) {
  5795. const parts = /^\w+:(.+)/.exec(val);
  5796. if (parts) {
  5797. const bsFilter = loadFilters();
  5798. const filterType = bsFilter.get(parts[1].trim().toLocaleLowerCase());
  5799. if (filterType) {
  5800. const { reg } = filterType;
  5801. for (const { name, users, _mujs } of cacheValues) {
  5802. if ([{ name }, ...users].find((o) => o.name.match(reg))) continue;
  5803. finds.add(_mujs.root);
  5804. }
  5805. }
  5806. }
  5807. } else if (val.match(/^recommend:/)) {
  5808. for (const { url, id, users, _mujs } of cacheValues) {
  5809. if (
  5810. users.find((u) => u.id === authorID) ||
  5811. goodUserJS.includes(url) ||
  5812. goodUserJS.includes(id)
  5813. ) {
  5814. finds.add(_mujs.root);
  5815. }
  5816. }
  5817. } else {
  5818. const reg = new RegExp(val, 'gi');
  5819. for (const v of cacheValues) {
  5820. if (v.name && v.name.match(reg)) finds.add(v._mujs.root);
  5821. if (v.description && v.description.match(reg)) finds.add(v._mujs.root);
  5822. if (v._mujs.code.data_meta)
  5823. for (const key of Object.keys(v._mujs.code.data_meta))
  5824. if (/name|desc/i.test(key) && key.match(reg)) finds.add(v._mujs.root);
  5825. }
  5826. }
  5827. dom.cl.add(qsA('tr[data-engine]', tabbody), 'hidden');
  5828. dom.cl.remove([...finds], 'hidden');
  5829. });
  5830. ael(urlBar, 'change', (evt) => {
  5831. evt.preventDefault();
  5832. const val = evt.target.value;
  5833. const tabElem = Tabs.getActive();
  5834. if (urlBar.placeholder === i18n$('newTab') && tabElem) {
  5835. const tabHost = tabElem.firstElementChild;
  5836. const host = formatURL(normalizedHostname(val));
  5837. if (Tabs.protoReg.test(val)) {
  5838. const createdTab = Tabs.getTab(val);
  5839. Tabs.close(tabElem);
  5840. if (createdTab) {
  5841. Tabs.active(createdTab);
  5842. } else {
  5843. Tabs.create(val);
  5844. }
  5845. evt.target.placeholder = i18n$('search_placeholder');
  5846. evt.target.value = '';
  5847. } else if (host === '*') {
  5848. tabElem.dataset.host = host;
  5849. tabHost.title = '<All Sites>';
  5850. tabHost.textContent = '<All Sites>';
  5851. MUList.host = host;
  5852. respHandles.build();
  5853. } else if (container.checkBlacklist(host)) {
  5854. showError(`Blacklisted "${host}"`);
  5855. } else {
  5856. tabElem.dataset.host = host;
  5857. tabHost.title = host;
  5858. tabHost.textContent = host;
  5859. MUList.host = host;
  5860. respHandles.build();
  5861. }
  5862. }
  5863. });
  5864. scheduler.postTask(makecfg, { priority: 'background' });
  5865. respHandles.build = async () => {
  5866. await scheduler.postTask(MUList.build, { priority: 'background' });
  5867. container.timeoutFrame();
  5868. };
  5869. if (cfg.autofetch) {
  5870. respHandles.build();
  5871. } else {
  5872. container.timeoutFrame();
  5873. }
  5874. } catch (ex) {
  5875. err(ex);
  5876. container.remove();
  5877. }
  5878. return respHandles;
  5879. }
  5880. // #endregion
  5881. /**
  5882. * @template { Function } F
  5883. * @param { (this: F, doc: Document) => * } onDomReady
  5884. */
  5885. const loadDOM = (onDomReady) => {
  5886. if (isFN(onDomReady)) {
  5887. if (document.readyState === 'interactive' || document.readyState === 'complete') {
  5888. onDomReady(document);
  5889. } else {
  5890. document.addEventListener('DOMContentLoaded', (evt) => onDomReady(evt.target), {
  5891. once: true
  5892. });
  5893. }
  5894. }
  5895. };
  5896. const init = async (prefix = 'Config') => {
  5897. const stored = await StorageSystem.getValue(prefix, DEFAULT_CONFIG);
  5898. cfg = {
  5899. ...DEFAULT_CONFIG,
  5900. ...stored
  5901. };
  5902. info('Config:', cfg);
  5903. loadDOM((doc) => {
  5904. try {
  5905. if (window.location === null)
  5906. throw new Error('"window.location" is null, reload the webpage or use a different one', {
  5907. cause: 'loadDOM'
  5908. });
  5909. if (doc === null)
  5910. throw new Error('"doc" is null, reload the webpage or use a different one', {
  5911. cause: 'loadDOM'
  5912. });
  5913. container.redirect();
  5914. if (cfg.autoinject) {
  5915. container.inject(primaryFN, doc);
  5916. } else {
  5917. container.timeoutFrame();
  5918. }
  5919. Command.register(i18n$('userjs_inject'), () => {
  5920. container.inject(primaryFN, doc);
  5921. });
  5922. Command.register(i18n$('userjs_close'), () => {
  5923. container.remove();
  5924. });
  5925. } catch (ex) {
  5926. err(ex);
  5927. }
  5928. });
  5929. };
  5930. init();
  5931. })();