GreasyFork Bullshit Filter

Hides scripts for popular browser games and social networks as well as scripts that use "foreign" characters in descriptions. Applies to posts in Forum too.

  1. // ==UserScript==
  2. // @name GreasyFork Bullshit Filter
  3. // @namespace darkred
  4. // @version 2021.3.31
  5. // @description Hides scripts for popular browser games and social networks as well as scripts that use "foreign" characters in descriptions. Applies to posts in Forum too.
  6. // @author kuehlschrank, darkred, valacar, Graphen
  7. // @license MIT
  8. // @icon https://raw.githubusercontent.com/darkred/Userscripts/master/GreasyFork_Bullshit_Filter/large.png
  9. // @include /^https:\/\/(greasy|sleazy)fork\.org\/(.*\/)?(scripts|discussions|users).*$/
  10. // @exclude /^https:\/\/(greasy|sleazy)fork\.org\/(.*\/)?(scripts\/[\w-]+\/feedback|discussions\/[\d]+|users\/.*\/conversations.*)$/
  11. // @grant none
  12. // This is a modified version of this script (http://userscripts-mirror.org/scripts/show/97145) by kuehlschrank.
  13. // Thanks a lot to:
  14. // - kuehlschrank for making another great script,
  15. // - valacar for the refactoring,
  16. // - Graphen for the 'Non-Latin' regex.
  17. // @supportURL https://github.com/darkred/Userscripts/issues
  18. // ==/UserScript==
  19.  
  20.  
  21. (function() {
  22.  
  23. const DEBUGGING = 0;
  24.  
  25. const filters = {
  26. 'Non-ASCII': /[^\x00-\x7F\s]+/,
  27. 'Non-Latin': /[^\u0000-\u024F\u2000-\u214F\s]+/,
  28. 'Games': /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*Tanks|West\s*Wars|wings\.io|\bWoD\b|World\s*of\s*Dungeons|wtf\s*battles|Wurzelimperium|Yohoho|Zombs/iu,
  29. 'Social Networks': /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/iu,
  30. 'Clutter': /^\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)?/iu
  31. };
  32.  
  33. const commonCss = `
  34. .filter-status {
  35. margin-left: 6px;
  36. }
  37.  
  38. .filter-switches {
  39. display: none;
  40. }
  41.  
  42. :hover>.filter-switches {
  43. display: block;
  44. }
  45.  
  46. .filter-on,
  47. .filter-off {
  48. display: block;
  49. width: 100px;
  50. }
  51.  
  52. .filter-switches a {
  53. text-decoration: none;
  54. color: inherit;
  55. cursor: pointer;
  56. }
  57.  
  58. .filter-switches a {
  59. margin-left: 8px;
  60. padding: 0 4px;
  61. }
  62.  
  63. a.filter-on {
  64. background-color: #ffcccc;
  65. color: #333333;
  66. text-decoration: line-through;
  67. }
  68.  
  69. a.filter-off {
  70. background-color: #ccffcc;
  71. color: #333333;
  72. }
  73. `;
  74.  
  75. const isOnForum = /discussions|feedback/.test(window.location.href);
  76.  
  77. const site = {};
  78. if (isOnForum) {
  79. site.css = '.discussion-list-item.filtered { display: none; } .filter-on, .filter-off { color: black; } ' + commonCss;
  80. site.cssDebug = '.discussion-list-item.filtered { background-color: khaki; } ' + commonCss;
  81. site.filterStatusLocation = '.list-option-groups';
  82. site.itemsToCheck = '.discussion-list-item';
  83. site.itemType = 'discussions';
  84. site.removeFilter = function(el) {
  85. el.classList.remove('filtered');
  86. };
  87. site.applyFilter = function(el, activeFilter) {
  88. const temp = el.children[1].innerText;
  89. if (temp && temp.match(activeFilter)) {
  90. el.classList.add('filtered');
  91. return true;
  92. }
  93. return false;
  94. };
  95. } else {
  96. site.css = 'li.filtered { display: none; } ' + commonCss;
  97. site.cssDebug = 'li.filtered { background-color: khaki; } ' + commonCss;
  98. site.filterStatusLocation = '#script-list-option-groups';
  99. site.itemsToCheck = 'article > h2';
  100. site.itemType = 'scripts';
  101. site.removeFilter = function(el) {
  102. el.parentNode.parentNode.classList.remove('filtered');
  103. };
  104. site.applyFilter = function(el, activeFilter) {
  105. if (el.innerText.match(activeFilter)) {
  106. el.parentNode.parentNode.classList.add('filtered');
  107. return true;
  108. }
  109. return false;
  110. };
  111. }
  112.  
  113. insertStyle();
  114. insertStatus();
  115. filterScripts();
  116. insertSwitches();
  117.  
  118. function insertStyle() {
  119. const style = document.createElement('style');
  120. style.textContent = DEBUGGING ? site.cssDebug : site.css;
  121. style.type = 'text/css';
  122. document.head.appendChild(style);
  123. }
  124.  
  125. function insertStatus() {
  126. const p = document.querySelector(site.filterStatusLocation);
  127. if (p) {
  128. const status = document.createElement('span');
  129. status.className = 'filter-status';
  130. p.appendChild(status);
  131. }
  132. }
  133.  
  134. function filterScripts() {
  135. const activeFilters = [];
  136. for (let filterType of Object.keys(filters)) {
  137. if (configGetValue(filterType, 'on') === 'on') {
  138. activeFilters.push(filters[filterType]);
  139. }
  140. }
  141. const nodes = document.querySelectorAll(site.itemsToCheck);
  142. let numFiltered = 0;
  143. for (let node of nodes) {
  144. site.removeFilter(node);
  145. for (let activeFilter of activeFilters) {
  146. let filtered = site.applyFilter(node, activeFilter);
  147. if (filtered) {
  148. numFiltered++;
  149. break;
  150. }
  151. }
  152. }
  153. const filterStatus = document.querySelector('.filter-status');
  154. if (filterStatus) {
  155. const numUnfiltered = document.querySelectorAll(site.itemsToCheck).length - numFiltered;
  156. filterStatus.textContent = `${numUnfiltered} ${site.itemType} (${numFiltered} filtered)`;
  157. }
  158. }
  159.  
  160. function insertSwitches() {
  161. const span = document.createElement('span');
  162. span.className = 'filter-switches';
  163. for (let filterType of Object.keys(filters)) {
  164. span.appendChild(createSwitch(filterType, configGetValue(filterType, 'on') === 'on'));
  165. }
  166. const filterStatus = document.querySelector('.filter-status');
  167. if (filterStatus) {
  168. filterStatus.parentNode.appendChild(span);
  169. }
  170. }
  171.  
  172. function createSwitch(label, isOn) {
  173. const a = document.createElement('a');
  174. a.className = isOn ? 'filter-on' : 'filter-off';
  175. a.textContent = label;
  176. a.addEventListener('click', function(e) {
  177. if (this.className === 'filter-on') {
  178. this.className = 'filter-off';
  179. configSetValue(this.textContent, 'off');
  180. } else {
  181. this.className = 'filter-on';
  182. configSetValue(this.textContent, 'on');
  183. }
  184. filterScripts();
  185. e.preventDefault();
  186. }, false);
  187. return a;
  188. }
  189.  
  190. function configSetValue(name, value) {
  191. localStorage.setItem(name, value);
  192. }
  193.  
  194. function configGetValue(name, defaultValue) {
  195. const value = localStorage.getItem(name);
  196. return value ? value : defaultValue;
  197. }
  198.  
  199. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址