Script Finder 油猴脚本查找

Script Finder 在任何网站上找到适用于该网站的greasyfork油猴脚本

当前为 2023-07-31 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Script Finder
  3. // @name:zh-CN Script Finder 油猴脚本查找
  4. // @namespace http://tampermonkey.net/
  5. // @version 0.1.3
  6. // @description Script Finder allows you to find userscripts from greasyfork on any website.
  7. // @description:zh-CN Script Finder 在任何网站上找到适用于该网站的greasyfork油猴脚本
  8. // @author shiquda
  9. // @namespace https://github.com/shiquda/shiquda_UserScript
  10. // @supportURL https://github.com/shiquda/shiquda_UserScript/issues
  11. // @match *://*/*
  12. // @icon 
  13. // @grant GM_xmlhttpRequest
  14. // @grant GM_addStyle
  15. // @license AGPL-3.0
  16. // ==/UserScript==
  17.  
  18. (function () {
  19. const domainParts = window.location.hostname.split('.').slice(-2);
  20. const domain = domainParts.join('.');
  21. const errorMessage = "Failed to retrieve script information or there are no available scripts for this domain.";
  22.  
  23. function getScriptsInfo(domain) {
  24.  
  25. var url = `https://greasyfork.org/scripts/by-site/${domain}?filter_locale=0`;
  26. GM_xmlhttpRequest({
  27. method: "GET",
  28. url: url,
  29. onload: (response) => {
  30.  
  31. const parser = new DOMParser();
  32. const doc = parser.parseFromString(response.responseText, "text/html");
  33. const scripts = doc.querySelector("#browse-script-list")?.querySelectorAll('[data-script-id]');
  34. let scriptsInfo = [];
  35. if (!scripts) {
  36. scriptsInfo = errorMessage
  37. } else {
  38. for (var i = 0; i < scripts.length; i++) {
  39. scriptsInfo.push(parseScriptInfo(scripts[i]));
  40. }
  41. }
  42. console.log(scriptsInfo);
  43. showScriptsInfo(scriptsInfo); // 显示脚本信息
  44. },
  45. onerror: () => {
  46. console.log("Some error!");
  47. const scriptsInfo = errorMessage
  48. showScriptsInfo(scriptsInfo)
  49. }
  50. });
  51. // oneClickInstall();
  52. }
  53.  
  54. // 解析脚本信息
  55. function parseScriptInfo(script) {
  56. return {
  57. id: script.getAttribute('data-script-id'),
  58. name: script.getAttribute('data-script-name'),
  59. author: script.querySelector("dd.script-list-author").textContent,
  60. description: script.querySelector(".script-description").textContent,
  61. version: script.getAttribute('data-script-version'),
  62. url: 'https://greasyfork.org/scripts/' + script.getAttribute('data-script-id'),
  63. createDate: script.getAttribute('data-script-created-date'),
  64. updateDate: script.getAttribute('data-script-updated-date'),
  65. installs: script.getAttribute('data-script-total-installs'),
  66. dailyInstalls: script.getAttribute('data-script-daily-installs'),
  67. ratingScore: script.getAttribute('data-script-rating-score')
  68. };
  69. }
  70.  
  71. function showScriptsInfo(scriptsInfo) {
  72. GM_addStyle(`
  73. button.script-button {
  74. position: fixed;
  75. bottom: 50%;
  76. right: -50px;
  77. transform: translateY(50%);
  78. padding: 10px;
  79. font-size: 16px;
  80. border: none;
  81. border-radius: 4px;
  82. background-color: #1e90ff;
  83. color: #ffffff;
  84. cursor: pointer;
  85. transition: right 0.3s;
  86. }
  87. div.info-container {
  88. display: none;
  89. position: fixed;
  90. top: 10%;
  91. right: 100px;
  92. width: 650px;
  93. padding: 12px;
  94. background-color: #ffffff;
  95. box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3);
  96. border-radius: 4px;
  97. opacity: 0;
  98. transition: opacity 0.3s;
  99. z-index: 9999;
  100. max-height: 80vh;
  101. overflow-y: auto;
  102. }
  103. ul.info-list {
  104. list-style: none;
  105. margin: 0;
  106. padding: 0;
  107. }
  108.  
  109. li.info-item {
  110. margin-bottom: 15px;
  111. padding: 12px;
  112. padding-bottom: 22px;
  113. display: flex;
  114. flex-direction: column;
  115. border: 1px solid #1e90ff;
  116. border-radius: 5px;
  117. }
  118.  
  119. .div.script-container {
  120. display: flex;
  121. flex-direction: column;
  122. }
  123.  
  124. a.script-link {
  125. font-size: 18px !important;
  126. font-weight: bold !important;
  127. margin-bottom: 5px !important;
  128. color: #1e90ff !important;
  129. }
  130.  
  131. p.script-description {
  132. color: black !important;
  133. margin-top: 2px;
  134. margin-bottom: 5px;
  135. font-size: 16px;
  136. }
  137.  
  138. div.details-container {
  139. font-size: 15px;
  140. font-weight: bold;
  141. display: flex;
  142. justify-content: space-between;
  143. margin-bottom: 15px;
  144. }
  145.  
  146. span.script-details {
  147. font: !important
  148. color: black !important;
  149. flex-grow: 1 !important;
  150. text-align: center !important;
  151. border: 1px solid #1e90ff !important;
  152. border-radius: 5px !important;
  153. margin: 4px !important;
  154. }
  155.  
  156. div.table-header {
  157. color: #1e90ff !important;
  158. font-size: 25px;
  159. font-weight: bold;
  160. }
  161.  
  162. input.script-search-input {
  163. width: 96% !important;
  164. padding: 10px !important;
  165. font-size: 18px !important;
  166. border: 1px solid #1e90ff !important;
  167. border-radius: 4px !important;
  168. box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3) !important;
  169. margin-bottom: 15px !important;
  170. margin-top: 20px !important;
  171. }
  172.  
  173. a.install-button {
  174. font-size: 20px;
  175. background-color: green;
  176. color: white;
  177. padding: 12px;
  178. }
  179.  
  180. button.to-greasyfork {
  181. position: absolute;
  182. top: 12px;
  183. right: 12px;
  184. border-radius: 4px;
  185. padding: 8px;
  186. font-size: 16px;
  187. border: none;
  188. background-color: #1e90ff;
  189. color: #ffffff;
  190. cursor: pointer;
  191. }
  192. `);
  193.  
  194.  
  195. // 创建打开列表按钮
  196. var button = document.createElement('button');
  197. button.className = 'script-button';
  198. button.innerText = 'Scripts';
  199.  
  200. // 创建脚本容器
  201. var infoContainer = document.createElement('div');
  202. infoContainer.className = 'info-container';
  203.  
  204. // 创建搜索框
  205. var searchInput = document.createElement('input');
  206. searchInput.type = 'text';
  207. searchInput.placeholder = 'Search scripts...';
  208. searchInput.className = 'script-search-input';
  209.  
  210. // 创建指向greasyfork的链接
  211.  
  212. var toGreasyfork = document.createElement('button');
  213. toGreasyfork.className = 'to-greasyfork';
  214. toGreasyfork.innerText = 'View on Greasyfork';
  215.  
  216. // 创建表头
  217. var tableHeader = document.createElement('div');
  218. tableHeader.className = 'table-header';
  219. tableHeader.appendChild(document.createTextNode('Script Finder'));
  220. tableHeader.appendChild(searchInput);
  221. tableHeader.appendChild(toGreasyfork);
  222.  
  223. // 创建脚本列表
  224. var infoList = document.createElement('ul');
  225. infoList.className = 'info-list';
  226.  
  227. // 插入脚本
  228. if (scriptsInfo === errorMessage) {
  229. infoList.innerHTML = errorMessage;
  230. } else {
  231. for (var i = 0; i < scriptsInfo.length; i++) {
  232. var script = scriptsInfo[i];
  233. var listItem = document.createElement('li');
  234. listItem.className = 'info-item';
  235.  
  236. var scriptContainer = document.createElement('div');
  237. scriptContainer.className = 'script-container';
  238.  
  239. var nameElement = document.createElement('a');
  240. nameElement.className = 'script-link';
  241. nameElement.innerText = script.name;
  242. nameElement.href = script.url;
  243. nameElement.target = '_blank';
  244.  
  245. var descriptionElement = document.createElement('p');
  246. descriptionElement.className = 'script-description';
  247. descriptionElement.innerHTML = script.description;
  248.  
  249. var detailsContainer = document.createElement('div');
  250. detailsContainer.className = 'details-container';
  251.  
  252. // 创建一键安装按钮
  253. var installButton = document.createElement('a');
  254. installButton.className = 'install-button';
  255. installButton.innerText = `Install ${script.version}`;
  256. installButton.href = `https://greasyfork.org/scripts/${script.id}/code/script.user.js`;
  257.  
  258. const details = [
  259. { key: 'Author', value: script.author },
  260. { key: 'Installs', value: script.installs },
  261. { key: 'Daily Installs', value: script.dailyInstalls },
  262. { key: 'Created', value: script.createDate },
  263. { key: 'Updated', value: script.updateDate },
  264. { key: 'Rating', value: script.ratingScore }
  265. ];
  266.  
  267. for (let i = 0; i < details.length; i++) {
  268. const spanElement = document.createElement('span');
  269. spanElement.className = 'script-details';
  270. spanElement.innerText = `${details[i].key}:\n${details[i].value}`;
  271. detailsContainer.appendChild(spanElement);
  272. }
  273.  
  274. scriptContainer.appendChild(nameElement);
  275. scriptContainer.appendChild(descriptionElement);
  276. scriptContainer.appendChild(detailsContainer);
  277. scriptContainer.appendChild(installButton);
  278.  
  279. listItem.appendChild(scriptContainer);
  280. listItem.scriptId = script.id;
  281. infoList.appendChild(listItem);
  282. }
  283. }
  284.  
  285. infoContainer.appendChild(tableHeader)
  286.  
  287. infoContainer.appendChild(infoList);
  288.  
  289. var timeout;
  290. button.addEventListener('mouseenter', function () {
  291. clearTimeout(timeout);
  292. button.style.right = '10px';
  293. });
  294.  
  295. button.addEventListener('mouseleave', function () {
  296. timeout = setTimeout(function () {
  297. button.style.right = '-50px';
  298. }, 500);
  299. });
  300.  
  301. button.addEventListener('click', function (event) {
  302. event.stopPropagation();
  303. button.style.right = '10px';
  304. infoContainer.style.display = "block"
  305. infoContainer.style.opacity = 1
  306. });
  307.  
  308. infoContainer.addEventListener('click', function (event) {
  309. event.stopPropagation();
  310. });
  311.  
  312. searchInput.addEventListener('input', () => {
  313. searchScript()
  314. })
  315.  
  316. toGreasyfork.addEventListener('click', function () {
  317. window.open(`https://greasyfork.org/scripts/by-site/${domain}?q=${searchInput.value}`)
  318. })
  319. document.body.addEventListener('click', function () {
  320. clearTimeout(timeout);
  321. button.style.right = '-50px';
  322. infoContainer.style.opacity = 0
  323. infoContainer.style.display = "none"
  324. });
  325.  
  326. document.body.appendChild(button);
  327.  
  328. document.body.appendChild(infoContainer);
  329. }
  330.  
  331. function searchScript() {
  332. const searchWord = document.querySelector('.script-search-input').value.toLowerCase(); // 将要匹配的文本转换为小写
  333. const scriptList = document.querySelectorAll('.info-item');
  334. for (let i = 0; i < scriptList.length; i++) {
  335. const scriptText = scriptList[i].innerText.toLowerCase(); // 将检索的文本转换为小写
  336. if (scriptText.includes(searchWord)) {
  337. scriptList[i].style.display = 'block';
  338. } else {
  339. scriptList[i].style.display = 'none';
  340. }
  341. }
  342. }
  343.  
  344. getScriptsInfo(domain);
  345.  
  346. })();