Greasy Fork is available in English.

Script Finder Recherche de script GreasyFork

Script Finder Rechercher sur n’importe quel site Web qui s’applique à ce site Web Script de GreasyFork。

Installer ce script?
Script suggéré par l'auteur

Vous pourriez également aimer Greasyfork Artisan de la colle.

Installer ce script
  1. // ==UserScript==
  2. // @name Script Finder+
  3. // @name:zh-CN Script Finder 油猴脚本查找
  4. // @description:zh-CN Script Finder 在任何网站上找到适用于该网站的 油猴脚本。
  5. // @name:ar Script Finder البحث عن نص قرد الشحوم
  6. // @description:ar Script Finder ابحث في أي موقع ويب ينطبق على هذا الموقع سيناريو القرد الشحوم。
  7. // @name:bg Script Finder Търсене на скрипт на GreasyFork
  8. // @description:bg Script Finder Намерете във всеки уебсайт, който се отнася за този уебсайт GreasyFork Script。
  9. // @name:cs Script Finder Vyhledávání skriptů GreasyFork
  10. // @description:cs Script Finder Vyhledejte na libovolném webu, který se tohoto webu týká GreasyFork Script。
  11. // @name:da Script Finder GreasyFork Script-opslag
  12. // @description:da Script Finder Find på ethvert websted, der gælder for det pågældende websted GreasyFork Script。
  13. // @name:de Script Finder Nachschlagen von GreasyFork-Skripten
  14. // @description:de Script Finder Finden Sie auf jeder Website, die für diese Website gilt GreasyFork-Skript。
  15. // @name:el Script Finder Αναζήτηση σεναρίου GreasyFork
  16. // @description:el Script Finder Βρείτε σε οποιονδήποτε ιστότοπο που ισχύει για αυτόν τον ιστότοπο Σενάριο GreasyFork。
  17. // @name:en Script Finder GreasyFork Script Lookup
  18. // @description:en Script Finder Find on any website that applies to that website GreasyFork Script。
  19. // @name:eo Script Finder GreasyFork Skripto Serĉo
  20. // @description:eo Script Finder Trovu en iu ajn retejo kiu validas por tiu retejo GreasyFork Skripto。
  21. // @name:es Script Finder Búsqueda de guiones de GreasyFork
  22. // @description:es Script Finder Busque en cualquier sitio web que se aplique a ese sitio web. Guión del mono de grasa。
  23. // @name:fi Script Finder GreasyFork Script Lookup
  24. // @description:fi Script Finder Etsi miltä tahansa verkkosivustoa, joka koskee kyseistä verkkosivustoa GreasyFork Script。
  25. // @name:fr Script Finder Recherche de script GreasyFork
  26. // @description:fr Script Finder Rechercher sur n’importe quel site Web qui s’applique à ce site Web Script de GreasyFork。
  27. // @name:he Script Finder בדיקת סקריפט של גריז קוף
  28. // @description:he Script Finder מצא בכל אתר הרלוונטי לאותו אתר סקריפט גריז קוף。
  29. // @name:hr Script Finder Traženje skripte GreasyFork
  30. // @description:hr Script Finder Pronađite na bilo kojoj web stranici koja se odnosi na tu web stranicu GreasyFork Script。
  31. // @name:hu Script Finder GreasyFork Script Lookup
  32. // @description:hu Script Finder Keresse meg az adott webhelyre vonatkozó bármely webhelyet GreasyFork Script。
  33. // @name:id Script Finder Pencarian Skrip GreasyFork
  34. // @description:id Script Finder Temukan di situs web mana pun yang berlaku untuk situs web tersebut Naskah Monyet Gemuk。
  35. // @name:it Script Finder Ricerca script GreasyFork
  36. // @description:it Script Finder Trova su qualsiasi sito Web che si applica a quel sito Web Sceneggiatura della scimmia grassa。
  37. // @name:ja Script Finder GreasyFork スクリプトの検索
  38. // @description:ja Script Finder その Web サイトに該当する Web サイトを検索する グリース モンキー スクリプト。
  39. // @name:ka Script Finder GreasyFork სკრიპტის ძიება
  40. // @description:ka Script Finder იპოვეთ ნებისმიერ ვებსაიტზე, რომელიც ეხება ამ ვებსაიტს GreasyFork Script。
  41. // @name:ko Script Finder 그리스 원숭이 스크립트 조회
  42. // @description:ko Script Finder 해당 웹사이트에 적용되는 웹사이트를 찾으세요. 그리스 원숭이 스크립트。
  43. // @name:nl Script Finder GreasyFork-script opzoeken
  44. // @description:nl Script Finder Zoek op elke website wat op die website van toepassing is GreasyFork-script。
  45. // @name:nb Script Finder GreasyFork Script Lookup
  46. // @description:nb Script Finder Finn på et hvilket som helst nettsted som gjelder for det nettstedet GreasyFork Script。
  47. // @name:pl Script Finder Wyszukiwanie skryptu GreasyFork
  48. // @description:pl Script Finder Znajdź na dowolnej stronie internetowej, która dotyczy tej witryny Smaruj skrypt małpy。
  49. // @name:pt-BR Script Finder Pesquisa de script do GreasyFork
  50. // @description:pt-BR Script Finder Encontre em qualquer site que se aplique a esse site Script do Macaco Graxa。
  51. // @name:ro Script Finder Căutare Script GreasyFork
  52. // @description:ro Script Finder Găsiți pe orice site care se aplică acelui site GreasyFork Script。
  53. // @name:ru Script Finder Поиск сценария GreasyFork
  54. // @description:ru Script Finder Найти на любом веб-сайте, который относится к этому веб-сайту Сценарий GreasyFork。
  55. // @name:sk Script Finder Vyhľadávanie skriptov GreasyFork
  56. // @description:sk Script Finder Nájdite na ľubovoľnej webovej lokalite, ktorá sa týka danej webovej lokality GreasyFork Script。
  57. // @name:sr Script Finder Греасе Монкеи Сцрипт Лоокуп
  58. // @description:sr Script Finder Пронађите на било којој веб локацији која се односи на ту веб локацију Греасе Монкеи Сцрипт。
  59. // @name:sv Script Finder GreasyFork Script Lookup
  60. // @description:sv Script Finder Hitta på vilken webbplats som helst som gäller den webbplatsen GreasyFork Script。
  61. // @name:th Script Finder ค้นหาสคริปต์ GreasyFork
  62. // @description:th Script Finder ค้นหาบนเว็บไซต์ใด ๆ ที่ใช้กับเว็บไซต์นั้น สคริปต์ลิงจาระบี。
  63. // @name:tr Script Finder GreasyFork Komut Dosyası Arama
  64. // @description:tr Script Finder Söz konusu web sitesi için geçerli olan herhangi bir web sitesinde bulun GreasyFork Komut Dosyası。
  65. // @name:ug Script Finder مايمۇن قوليازمىسىنى ئىزدەش
  66. // @description:ug Script Finder شۇ تور بېكەتكە ماس كېلىدىغان ھەرقانداق تور بەتنى ئىزدەڭ مايمۇن قوليازمىسى。
  67. // @name:uk Script Finder Пошук сценарію GreasyFork
  68. // @description:uk Script Finder Знайдіть на будь-якому веб-сайті, який стосується цього веб-сайту Сценарій GreasyFork。
  69. // @name:vi Script Finder Tra cứu tập lệnh GreasyFork
  70. // @description:vi Script Finder Tìm trên bất kỳ trang web nào áp dụng cho trang web đó Kịch bản khỉ mỡ。
  71. // @name:zh-TW Script Finder 油猴腳本查找
  72. // @description:zh-TW Script Finder 在任何網站上找到適用於該網站的 油猴腳本。
  73. // @name:zh-HK Script Finder 油猴腳本查找
  74. // @description:zh-HK Script Finder 在任何網站上找到適用於該網站的 油猴腳本。
  75. // @name:fr-CA Script Finder Recherche de script GreasyFork
  76. // @description:fr-CA Script Finder Rechercher sur n’importe quel site Web qui s’applique à ce site Web Script de GreasyFork。
  77. // @description Script Finder allows you to find userscripts from greasyfork on any website.
  78.  
  79. // @namespace https://github.com/ChinaGodMan/UserScripts
  80. // @version 0.1.6.80
  81. // @author shiquda & 人民的勤务员 <china.qinwuyuan@gmail.com>
  82. // @supportURL https://github.com/ChinaGodMan/UserScripts/issues
  83. // @homepageURL https://github.com/ChinaGodMan/UserScripts
  84. // @match *://*/*
  85. // @connect greasyfork.org
  86. // @icon 
  87. // @iconbak https://github.com/ChinaGodMan/UserScripts/raw/main/docs/icon/Scripts%20Icons/Finder.jpg
  88. // @grant GM_xmlhttpRequest
  89. // @grant GM_addStyle
  90. // @license MIT
  91.  
  92. // ==/UserScript==
  93. const translate = (function () {
  94. const userLang = (navigator.languages && navigator.languages[0]) || navigator.language || 'en'
  95. const strings = {
  96. 'en': {
  97. Author: 'Author',
  98. Installs: 'Installs',
  99. DailyInstalls: 'Daily Installs',
  100. Created: 'Created',
  101. Updated: 'Updated',
  102. Rating: 'Rating',
  103. LoadingScripts: 'Loading scripts...',
  104. LoadMore: 'Load more',
  105. AllScriptsLoaded: 'All scripts loaded',
  106. SearchPlaceholder: 'Search scripts...',
  107. ViewOnGreasyfork: 'View on Greasyfork',
  108. errorMessage: 'Failed to retrieve script information or there are no available scripts for this domain.',
  109. Loading: 'Loading...',
  110. Scripts: 'Scripts'
  111. },
  112. 'zh-CN': {
  113. Author: '作者',
  114. Installs: '安装数量',
  115. DailyInstalls: '每日安装',
  116. Created: '创建日期',
  117. Updated: '更新时间',
  118. Rating: '评分',
  119. LoadingScripts: '正在加载脚本...',
  120. LoadMore: '加载更多',
  121. AllScriptsLoaded: '所有脚本已加载',
  122. SearchPlaceholder: '搜索脚本...',
  123. ViewOnGreasyfork: '在Greasyfork查看',
  124. errorMessage: '无法检索脚本信息或该域没有可用的脚本。',
  125. Loading: '载入中...',
  126. Scripts: '脚本'
  127.  
  128. },
  129. 'zh-TW': {
  130. Author: '作者',
  131. Installs: '安裝數量',
  132. DailyInstalls: '每日安裝',
  133. Created: '創建日期',
  134. Updated: '更新日期',
  135. Rating: '評分',
  136. LoadingScripts: '正在載入腳本...',
  137. LoadMore: '載入更多',
  138. AllScriptsLoaded: '所有腳本已載入',
  139. SearchPlaceholder: '搜尋腳本...',
  140. ViewOnGreasyfork: '在Greasyfork查看',
  141. errorMessage: '無法檢索腳本信息或該域沒有可用的腳本。',
  142. Loading: '載入中...',
  143. Scripts: '腳本'
  144. },
  145. 'ja': {
  146. Author: '著者',
  147. Installs: 'インストール数',
  148. DailyInstalls: '日次インストール数',
  149. Created: '作成日',
  150. Updated: '更新日',
  151. Rating: '評価',
  152. LoadingScripts: 'スクリプトを読み込んでいます...',
  153. LoadMore: 'もっと読む',
  154. AllScriptsLoaded: 'すべてのスクリプトが読み込まれました',
  155. SearchPlaceholder: 'スクリプトを検索...',
  156. ViewOnGreasyfork: 'Greasyforkで見る',
  157. errorMessage: 'スクリプト情報の取得に失敗するか、またはこのドメインには利用可能なスクリプトがありません。',
  158. Loading: '読み込み中...',
  159. Scripts: 'スクリプト'
  160. },
  161. 'vi': {
  162. Author: 'Tác giả',
  163. Installs: 'Số lượt cài đặt',
  164. DailyInstalls: 'Cài đặt hàng ngày',
  165. Created: 'Ngày tạo',
  166. Updated: 'Ngày cập nhật',
  167. Rating: 'Đánh giá',
  168. LoadingScripts: 'Đang tải các tập lệnh...',
  169. LoadMore: 'Tải thêm',
  170. AllScriptsLoaded: 'Đã tải tất cả các tập lệnh',
  171. SearchPlaceholder: 'Tìm kiếm tập lệnh...',
  172. ViewOnGreasyfork: 'Xem trên Greasyfork',
  173. errorMessage: 'Không thể truy xuất thông tin tập lệnh hoặc không có tập lệnh nào có sẵn cho miền này.',
  174. Loading: 'Đang tải...',
  175. Scripts: 'Tập lệnh'
  176. }
  177. }
  178. // 返回翻译函数
  179. return (id, lang = '') => {
  180. const selectedLang = lang || userLang
  181. return (strings[selectedLang] || strings.en)[id] || strings.en[id]
  182. }
  183. }());
  184. (function () {
  185. const domainParts = window.location.hostname.split('.').slice(-2)
  186. const domain = domainParts.join('.')
  187. const errorMessage = translate('errorMessage')
  188. let neverLoadedScripts = true
  189. let collapsed = true
  190. let loadedPages = 0
  191.  
  192. function getScriptsInfo(domain, page = 1) {
  193. var url = `https://greasyfork.org/scripts/by-site/${domain}?filter_locale=0&sort=updated&page=${page}`
  194.  
  195. GM_xmlhttpRequest({
  196. method: 'GET',
  197. url: url,
  198. onload: (response) => {
  199. // 解析结果
  200. const parser = new DOMParser()
  201. const doc = parser.parseFromString(response.responseText, 'text/html')
  202. const scripts = doc.querySelector('#browse-script-list')?.querySelectorAll('[data-script-id]')
  203. let scriptsInfo = []
  204.  
  205. if (!scripts) {
  206. scriptsInfo = errorMessage
  207. } else {
  208. for (var i = 0; i < scripts.length; i++) {
  209. scriptsInfo.push(parseScriptInfo(scripts[i]))
  210. }
  211. }
  212.  
  213. // 处理对象
  214. const loadMoreButton = document.querySelector('.load-more')
  215. console.log(doc.querySelector('.next_page'))
  216. if (doc.querySelector('.next_page') == null || doc.querySelector('.next_page')?.getAttribute('aria-disabled') === 'true') {
  217. loadedPages = 'max'
  218. loadMoreButton.disabled = true
  219. loadMoreButton.textContent = translate('AllScriptsLoaded')
  220. } else {
  221. loadMoreButton.disabled = false
  222. loadMoreButton.textContent = translate('LoadMore')
  223. }
  224. // console.log(scriptsInfo);
  225. document.querySelector('.wait-loading').style.display = 'none'
  226. loadMoreButton.style.display = 'block'
  227. appendScriptsInfo(scriptsInfo)
  228. updateMatches()
  229.  
  230. loadedPages = typeof loadedPages === 'number' ? loadedPages + 1 : 0
  231. // console.log(loadedPages)
  232. },
  233. onerror: () => {
  234. console.log('Some error occurred!')
  235. if (loadedPages === 0) {
  236. appendScriptsInfo(scriptsInfo)
  237. }
  238. const scriptsInfo = errorMessage
  239. document.querySelector('.wait-loading').style.display = 'none'
  240. }
  241. })
  242. }
  243.  
  244. // 解析脚本信息
  245. function parseScriptInfo(script) {
  246. return {
  247. id: script.getAttribute('data-script-id'),
  248. name: script.getAttribute('data-script-name'),
  249. author: script.querySelector('dd.script-list-author').textContent,
  250. description: script.querySelector('.script-description').textContent,
  251. version: script.getAttribute('data-script-version'),
  252. url: 'https://greasyfork.org/scripts/' + script.getAttribute('data-script-id'),
  253. createDate: script.getAttribute('data-script-created-date'),
  254. updateDate: script.getAttribute('data-script-updated-date'),
  255. installs: script.getAttribute('data-script-total-installs'),
  256. dailyInstalls: script.getAttribute('data-script-daily-installs'),
  257. ratingScore: script.getAttribute('data-script-rating-score')
  258. }
  259. }
  260.  
  261. // 插入脚本
  262. function appendScriptsInfo(scriptsInfo) {
  263. const infoList = document.querySelector('.info-list')
  264. if (scriptsInfo === errorMessage) {
  265. // infoList.innerHTML = errorMessage;
  266. const loadMoreButton = document.querySelector('.load-more')
  267. loadMoreButton.disabled = true
  268. loadMoreButton.textContent = translate('AllScriptsLoaded')
  269. loadMoreButton.innerHTML = errorMessage
  270. } else {
  271. for (var i = 0; i < scriptsInfo.length; i++) {
  272. var script = scriptsInfo[i]
  273. var listItem = document.createElement('li')
  274. listItem.className = 'info-item'
  275.  
  276. var scriptContainer = document.createElement('div')
  277. scriptContainer.className = 'script-container'
  278.  
  279. var nameElement = document.createElement('a')
  280. nameElement.className = 'mscript-link'
  281. nameElement.innerText = script.name
  282. nameElement.href = script.url
  283. nameElement.target = '_blank'
  284.  
  285. var descriptionElement = document.createElement('p')
  286. descriptionElement.className = 'script-description'
  287. descriptionElement.innerHTML = script.description
  288.  
  289. var detailsContainer = document.createElement('div')
  290. detailsContainer.className = 'details-container'
  291.  
  292. // 创建一键安装按钮
  293. var installButton = document.createElement('a')
  294. installButton.className = 'install-button'
  295. installButton.innerText = `Install ${script.version}`
  296. installButton.href = `https://greasyfork.org/scripts/${script.id}/code/script.user.js`
  297.  
  298. const details = [
  299. { key: translate('Author'), value: script.author },
  300. { key: translate('Installs'), value: script.installs },
  301. { key: translate('DailyInstalls'), value: script.dailyInstalls },
  302. { key: translate('Created'), value: script.createDate },
  303. { key: translate('Updated'), value: script.updateDate },
  304. { key: translate('Rating'), value: script.ratingScore }
  305. ]
  306.  
  307. for (let i = 0; i < details.length; i++) {
  308. const spanElement = document.createElement('span')
  309. spanElement.className = 'script-details'
  310. spanElement.innerText = `${details[i].key}:\n${details[i].value}`
  311. detailsContainer.appendChild(spanElement)
  312. }
  313.  
  314. scriptContainer.appendChild(nameElement)
  315. scriptContainer.appendChild(descriptionElement)
  316. scriptContainer.appendChild(detailsContainer)
  317. scriptContainer.appendChild(installButton)
  318.  
  319. listItem.appendChild(scriptContainer)
  320. listItem.scriptId = script.id
  321. infoList.appendChild(listItem)
  322. }
  323. }
  324. }
  325.  
  326. function setupUI() {
  327. GM_addStyle(`
  328. scrbutton.script-button {
  329. position: fixed;
  330. bottom: 20%;
  331. right: -50px;
  332. transform: translateY(50%);
  333. padding: 20px;
  334. font-size: 16px;
  335. border: none;
  336. border-radius: 4px;
  337. background-color: #1e90ff;
  338. color: #ffffff;
  339. cursor: pointer;
  340. transition: right 0.3s;
  341. z-index: 9999999999999999;
  342. }
  343. div.info-container {
  344. display: none;
  345. position: fixed;
  346. top: 50%;
  347. left: 50%;
  348. transform: translate(-50%, -50%);
  349. width: 650px;
  350. padding: 12px;
  351. background-color: #ffffff;
  352. box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3);
  353. border-radius: 4px;
  354. opacity: 0;
  355. transition: opacity 0.3s;
  356. z-index: 9999;
  357. max-height: 80vh;
  358. overflow-y: auto;
  359. }
  360. ul.info-list {
  361. list-style: none;
  362. margin: 0;
  363. padding: 0;
  364. }
  365. li.info-item {
  366. margin-bottom: 15px;
  367. padding: 12px;
  368. padding-bottom: 22px;
  369. display: flex;
  370. flex-direction: column;
  371. border: 1px solid #1e90ff;
  372. border-radius: 5px;
  373. }
  374. .div.script-container {
  375. display: flex;
  376. flex-direction: column;
  377. }
  378. a.mscript-link {
  379. font-size: 18px !important;
  380. font-weight: bold !important;
  381. margin-bottom: 5px !important;
  382. color: #1e90ff !important;
  383. }
  384. p.script-description {
  385. color: black !important;
  386. margin-top: 2px;
  387. margin-bottom: 5px;
  388. font-size: 16px;
  389. }
  390. div.details-container {
  391. font-size: 15px;
  392. font-weight: bold;
  393. display: flex;
  394. justify-content: space-between;
  395. margin-bottom: 15px;
  396. }
  397. span.script-details {
  398. font: !important;
  399. color: black !important;
  400. flex-grow: 1 !important;
  401. text-align: center !important;
  402. border: 1px solid #1e90ff !important;
  403. border-radius: 5px !important;
  404. margin: 4px !important;
  405. }
  406. div.table-header {
  407. color: #1e90ff !important;
  408. font-size: 25px;
  409. font-weight: bold;
  410. }
  411. input.script-search-input {
  412. width: 96% !important;
  413. padding: 10px !important;
  414. font-size: 18px !important;
  415. border: 1px solid #1e90ff !important;
  416. border-radius: 4px !important;
  417. box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3) !important;
  418. margin-bottom: 15px !important;
  419. margin-top: 20px !important;
  420. }
  421. a.install-button {
  422. font-size: 20px;
  423. background-color: green;
  424. color: white;
  425. padding: 12px;
  426. }
  427. button.to-greasyfork {
  428. position: absolute;
  429. top: 12px;
  430. right: 12px;
  431. border-radius: 4px;
  432. padding: 8px;
  433. font-size: 16px;
  434. border: none;
  435. background-color: #1e90ff;
  436. color: #ffffff;
  437. cursor: pointer;
  438. }
  439. span.match-count {
  440. background-color: #1e90ff;
  441. font-size: 25px;
  442. font-weight: bold;
  443. color: white;
  444. padding: 6px;
  445. position: absolute;
  446. right: 50%;
  447. border-radius: 12px;
  448. top: 10px;
  449. }
  450. div.wait-loading {
  451. font-size: 20px;
  452. font-weight: bold;
  453. color: #1e90ff;
  454. animation: blink 1s infinite;
  455. }
  456. @keyframes fadeInOut {
  457. 0% {
  458. opacity: 0;
  459. }
  460. 50% {
  461. opacity: 1;
  462. }
  463. 100% {
  464. opacity: 0;
  465. }
  466. }
  467. @keyframes blink {
  468. 0%, 100% {
  469. opacity: 0;
  470. }
  471. 50% {
  472. opacity: 1;
  473. }
  474. }
  475. button.load-more {
  476. border-radius: 4px;
  477. padding: 8px;
  478. font-size: 16px;
  479. border: none;
  480. background-color: #1e90ff;
  481. color: #ffffff;
  482. cursor: pointer;
  483. position: relative;
  484. bottom: 5px;
  485. left: 50%;
  486. transform: translateX(-50%);
  487. }
  488. button.load-more:disabled {
  489. background-color: #cccccc;
  490. cursor: not-allowed;
  491. }
  492.  
  493. /* Mobile styles */
  494. @media (max-width: 600px) {
  495. scrbutton.script-button {
  496. right: -30px;
  497. padding: 8px;
  498. font-size: 14px;
  499. }
  500. span.script-details {
  501. font-size: 10px !important;
  502. margin: 2px !important;
  503. padding: 2px !important;
  504. }
  505. span.match-count {
  506. font-size: 20px;
  507. padding: 4px;
  508. }
  509.  
  510. button.to-greasyfork {
  511. padding: 6px;
  512. font-size: 14px;
  513. }
  514.  
  515. a.install-button {
  516. font-size: 12px;
  517. padding: 8px;
  518. }
  519. div.table-header {
  520. font-size: 20px;
  521. }
  522. div.script-container {
  523. padding: 10px;
  524. }
  525. div.info-container {
  526. top: 10%;
  527. left: 5%;
  528. right: 5%;
  529. transform: none;
  530. width: calc(90% - 10px); /* 自适应宽度,保持左右边距 */
  531. max-width: 100%; /* 确保不超出屏幕宽度 */
  532. }
  533. a.mscript-link {
  534. font-size: 16px !important;
  535. }
  536. p.script-description {
  537. font-size: 14px;
  538. }
  539. {
  540.  
  541. input.script-search-input {
  542. width: 92% !important;
  543. padding: 8px !important;
  544. font-size: 16px !important;
  545. }
  546. span.match-count {
  547. font-size: 20px;
  548. padding: 4px;
  549. }
  550. button.load-more {
  551. font-size: 14px;
  552. padding: 6px;
  553. }
  554. }
  555. `)
  556.  
  557.  
  558. // 创建打开列表按钮
  559. var button = document.createElement('scrbutton')
  560. button.className = 'script-button'
  561. button.innerText = translate('Scripts')
  562. document.addEventListener('fullscreenchange', function () {
  563. if (document.fullscreenElement) {
  564. button.style.display = 'none'
  565. } else {
  566. button.style.display = 'block'
  567. }
  568. })
  569. // 创建脚本容器
  570. var infoContainer = document.createElement('div')
  571. infoContainer.className = 'info-container'
  572.  
  573. // 创建搜索框
  574. var searchInput = document.createElement('input')
  575. searchInput.type = 'text'
  576. searchInput.placeholder = translate('SearchPlaceholder')
  577. searchInput.className = 'script-search-input'
  578.  
  579. // 创建指向greasyfork的链接
  580. var toGreasyfork = document.createElement('button')
  581. toGreasyfork.className = 'to-greasyfork'
  582. toGreasyfork.innerText = translate('ViewOnGreasyfork')
  583.  
  584. // 创建计数器
  585. var matchCount = document.createElement('span')
  586. matchCount.className = 'match-count'
  587.  
  588. // 创建表头
  589. var tableHeader = document.createElement('div')
  590. tableHeader.className = 'table-header'
  591. tableHeader.appendChild(document.createTextNode('Script Finder'))
  592. tableHeader.appendChild(matchCount)
  593. tableHeader.appendChild(searchInput)
  594. tableHeader.appendChild(toGreasyfork)
  595.  
  596. // 创建脚本列表
  597. var infoList = document.createElement('ul')
  598. infoList.className = 'info-list'
  599.  
  600. // 创建等待加载
  601. var waitLoading = document.createElement('div')
  602. waitLoading.className = 'wait-loading'
  603. waitLoading.innerText = translate('LoadingScripts')
  604.  
  605. // 创建加载更多
  606. var loadMore = document.createElement('button')
  607. loadMore.className = 'load-more'
  608. loadMore.innerText = 'Load more'
  609. loadMore.style.display = 'none'
  610.  
  611. infoList.appendChild(waitLoading)
  612. infoList.appendChild(loadMore)
  613.  
  614. infoContainer.appendChild(tableHeader)
  615. infoContainer.appendChild(infoList)
  616.  
  617. var timeout
  618. button.addEventListener('mouseenter', function () {
  619. clearTimeout(timeout)
  620. button.style.right = '10px'
  621. })
  622.  
  623. button.addEventListener('mouseleave', function () {
  624. timeout = setTimeout(function () {
  625. button.style.right = '-50px'
  626. }, 500)
  627. })
  628.  
  629. button.addEventListener('click', function (event) {
  630. event.stopPropagation()
  631. if (collapsed) {
  632. infoContainer.style.display = 'block'
  633. infoContainer.style.opacity = 1
  634. collapsed = false
  635. }
  636. else {
  637. infoContainer.style.display = 'none'
  638. infoContainer.style.opacity = 0
  639. collapsed = true
  640. }
  641.  
  642. if (neverLoadedScripts) {
  643. getScriptsInfo(domain, 1)
  644. neverLoadedScripts = false
  645. }
  646.  
  647. })
  648.  
  649. infoContainer.addEventListener('click', function (event) {
  650. event.stopPropagation()
  651. })
  652.  
  653. searchInput.addEventListener('input', () => {
  654. searchScript()
  655. updateMatches()
  656. })
  657.  
  658. toGreasyfork.addEventListener('click', function () {
  659. window.open(`https://greasyfork.org/scripts/by-site/${domain}?q=${searchInput.value}&filter_locale=0&sort=updated`)
  660. })
  661.  
  662. loadMore.addEventListener('click', () => {
  663. if (loadedPages === 'max') {
  664. return
  665. }
  666. const loadMoreButton = document.querySelector('.load-more')
  667. loadMoreButton.disabled = true
  668. loadMoreButton.textContent = translate('Loading')
  669. document.querySelector('.wait-loading').style.display = 'block'
  670. getScriptsInfo(domain, loadedPages + 1)
  671. })
  672.  
  673. document.body.addEventListener('click', function () {
  674. clearTimeout(timeout)
  675. collapsed = true
  676. button.style.right = '-50px'
  677. infoContainer.style.opacity = 0
  678. infoContainer.style.display = 'none'
  679. })
  680.  
  681. document.body.appendChild(button)
  682.  
  683. document.body.appendChild(infoContainer)
  684.  
  685. infoContainer.addEventListener('change', () => {
  686. updateMatches()
  687. })
  688. updateMatches()
  689. }
  690.  
  691. function searchScript() {
  692. const searchWord = document.querySelector('.script-search-input').value.toLowerCase() // 将要匹配的文本转换为小写
  693. const scriptList = document.querySelectorAll('.info-item')
  694. for (let i = 0; i < scriptList.length; i++) {
  695. const scriptText = scriptList[i].innerText.toLowerCase() // 将检索的文本转换为小写
  696. if (scriptText.includes(searchWord)) {
  697. scriptList[i].style.display = 'block'
  698. } else {
  699. scriptList[i].style.display = 'none'
  700. }
  701. }
  702. }
  703.  
  704. function updateMatches() {
  705. const matchCount = document.querySelectorAll('.info-item:not([style*="display: none"])').length
  706. const allCount = document.querySelectorAll('.info-item').length
  707. document.querySelector('.match-count').innerText = matchCount === allCount ? matchCount : `${matchCount}/${allCount}`
  708. }
  709.  
  710. function main() {
  711. if (window.self !== window.top) {
  712. // 在iframe中执行时,直接退出
  713. return
  714. }
  715. setupUI()
  716. }
  717.  
  718. main()
  719.  
  720.  
  721. })()