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.

目前為 2018-09-20 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name GreasyFork Bullshit Filter
  3. // @namespace darkred
  4. // @version 2018.9.20
  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
  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|forum|users).*$/
  10. // @exclude /^https:\/\/(greasy|sleazy)fork\.org\/(.*\/)((scripts\/\d+)|forum\/(discussion\/|profile|messages)).*$/
  11. // @grant none
  12. // This is an edited version of this script (http://userscripts-mirror.org/scripts/show/97145) by kuehlschrank.
  13. // Thanks a lot to kuehlschrank for making another great script.
  14. // Thanks a lot to valacar for the refactoring.
  15. // ==/UserScript==
  16.  
  17.  
  18. (function() {
  19.  
  20. const DEBUGGING = 0;
  21.  
  22. const filters = {
  23. 'Non-ASCII': /[^\x00-\x7F\s]+/i,
  24. 'Games': /AntiGame|Agar|agar\.io|alis\.io|angel\.io|ExtencionRipXChetoMalo|AposBot|DFxLite|ZTx-Lite|AposFeedingBot|AposLoader|Blah Blah|Orc Clan Script|Astro\s*Empires|^\s*Attack|^\s*Battle|BiteFight|Blood\s*Wars|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|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|NeoQuest|MyFreeFarm|narwhale\.io|Neopets|Nemexia|\bOGame\b|Ogar(io)?|Pardus|Pennergame|Pigskin\s*Empire|PlayerScripts|pokeradar\.io|Popmundo|Po?we?r\s*(Bot|Tools)|PsicoTSI|Ravenwood|Schulterglatze|slither\.io|slitherplus\.io|slitheriogameplay|SpaceWars|splix\.io|\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/iu,
  25. '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,
  26. 'Clutter': /^\s*(.{1,3})\1+\n|^\s*(.+?)\n+\2\n*$|^\s*.{1,5}\n|do\s*n('|o)?t (install|download)|nicht installieren|(just)?(\s*a)?\s*test|^\s*.{0,4}test.{0,4}\n|\ntest(ing)?\s*|^\s*(\{@|Smolka|Hacks)|\[\d{4,5}\]|free\s*download|theme|(night|dark) ?(mode)?/iu
  27. };
  28.  
  29. const commonCss = `
  30. .filter-status {
  31. margin-left: 6px;
  32. }
  33.  
  34. .filter-switches {
  35. display: none;
  36. }
  37.  
  38. :hover>.filter-switches {
  39. display: block;
  40. }
  41.  
  42. .filter-on,
  43. .filter-off {
  44. display: block;
  45. width: 97px;
  46. }
  47.  
  48. .filter-switches a {
  49. text-decoration: none;
  50. color: inherit;
  51. cursor: pointer;
  52. }
  53.  
  54. .filter-switches a {
  55. margin-left: 8px;
  56. padding: 0 4px;
  57. }
  58.  
  59. a.filter-on {
  60. background-color: #ffcccc;
  61. color: #333333;
  62. text-decoration: line-through;
  63. }
  64.  
  65. a.filter-off {
  66. background-color: #ccffcc;
  67. color: #333333;
  68. }
  69. `;
  70.  
  71. const isOnForum = window.location.href.includes('forum');
  72.  
  73. const site = {};
  74. if (isOnForum) {
  75. site.css = '.ItemDiscussion.filtered { display: none; } .filter-on, .filter-off { color: black; } ' + commonCss;
  76. site.cssDebug = '.ItemDiscussion.filtered { background-color: khaki; } ' + commonCss;
  77. site.filterStatusLocation = '.PanelColumn';
  78. site.itemsToCheck = '.ItemDiscussion';
  79. site.itemType = 'discussions';
  80. site.removeFilter = function(el) {
  81. el.classList.remove('filtered');
  82. };
  83. site.applyFilter = function(el, activeFilter) {
  84. let temp = el.children[1].firstElementChild.innerText;
  85. if(temp && temp.match(activeFilter)) {
  86. el.classList.add('filtered');
  87. return true;
  88. }
  89. return false;
  90. };
  91. } else {
  92. site.css = 'li.filtered { display: none; } ' + commonCss;
  93. site.cssDebug = 'li.filtered { background-color: khaki; } ' + commonCss;
  94. site.filterStatusLocation = '#script-list-option-groups';
  95. site.itemsToCheck = 'article > h2';
  96. site.itemType = 'scripts';
  97. site.removeFilter = function(el) {
  98. el.parentNode.parentNode.classList.remove('filtered');
  99. };
  100. site.applyFilter = function(el, activeFilter) {
  101. if (el.innerText.match(activeFilter)) {
  102. el.parentNode.parentNode.classList.add('filtered');
  103. return true;
  104. }
  105. return false;
  106. };
  107. }
  108.  
  109. insertStyle();
  110. insertStatus();
  111. filterScripts();
  112. insertSwitches();
  113.  
  114. function insertStyle() {
  115. const style = document.createElement('style');
  116. style.textContent = DEBUGGING ? site.cssDebug : site.css;
  117. style.type = 'text/css';
  118. document.head.appendChild(style);
  119. }
  120.  
  121. function insertStatus() {
  122. const p = document.querySelector(site.filterStatusLocation);
  123. if (p) {
  124. const status = document.createElement('span');
  125. status.className = 'filter-status';
  126. p.appendChild(status);
  127. }
  128. }
  129.  
  130. function filterScripts() {
  131. const activeFilters = [];
  132. for (let filterType of Object.keys(filters)) {
  133. if (configGetValue(filterType, 'on') === 'on') {
  134. activeFilters.push(filters[filterType]);
  135. }
  136. }
  137. const nodes = document.querySelectorAll(site.itemsToCheck);
  138. let numFiltered = 0;
  139. for (let node of nodes) {
  140. site.removeFilter(node);
  141. for (let activeFilter of activeFilters) {
  142. let filtered = site.applyFilter(node, activeFilter);
  143. if (filtered) {
  144. numFiltered++;
  145. break;
  146. }
  147. }
  148. }
  149. const filterStatus = document.querySelector('.filter-status');
  150. if (filterStatus) {
  151. const numUnfiltered = document.querySelectorAll(site.itemsToCheck).length - numFiltered;
  152. filterStatus.textContent = `${numUnfiltered} ${site.itemType} (${numFiltered} filtered)`;
  153. }
  154. }
  155.  
  156. function insertSwitches() {
  157. const span = document.createElement('span');
  158. span.className = 'filter-switches';
  159. for (let filterType of Object.keys(filters)) {
  160. span.appendChild(createSwitch(filterType, configGetValue(filterType, 'on') === 'on'));
  161. }
  162. const filterStatus = document.querySelector('.filter-status');
  163. if (filterStatus) {
  164. filterStatus.parentNode.appendChild(span);
  165. }
  166. }
  167.  
  168. function createSwitch(label, isOn) {
  169. const a = document.createElement('a');
  170. a.className = isOn ? 'filter-on' : 'filter-off';
  171. a.textContent = label;
  172. a.addEventListener('click', function(e) {
  173. if (this.className === 'filter-on') {
  174. this.className = 'filter-off';
  175. configSetValue(this.textContent, 'off');
  176. } else {
  177. this.className = 'filter-on';
  178. configSetValue(this.textContent, 'on');
  179. }
  180. filterScripts();
  181. e.preventDefault();
  182. }, false);
  183. return a;
  184. }
  185.  
  186. function configSetValue(name, value) {
  187. localStorage.setItem(name, value);
  188. }
  189.  
  190. function configGetValue(name, defaultValue) {
  191. const value = localStorage.getItem(name);
  192. return value ? value : defaultValue;
  193. }
  194.  
  195. })();