// ==UserScript== // @name Script Finder+ // @name:zh-CN Script Finder 油猴脚本查找 // @namespace https://greasyfork.org/zh-CN/users/1169082 // @version 0.1.6.53 // @description Script Finder allows you to find userscripts from greasyfork on any website. // @description:zh-CN Script Finder 在任何网站上找到适用于该网站的greasyfork油猴脚本 // @author shiquda & 人民的勤务员 // @supportURL https://github.com/ChinaGodMan/UserScripts/issues // @homepageURL https://github.com/ChinaGodMan/UserScripts // @match *://*/* // @connect greasyfork.org // @icon https://github.com/ChinaGodMan/UserScripts/raw/main/docs/icon/Scripts%20Icons/Finder.jpg // @grant GM_xmlhttpRequest // @grant GM_addStyle // @license AGPL-3.0 // @downloadURL none // ==/UserScript== (function () { const domainParts = window.location.hostname.split('.').slice(-2) const domain = domainParts.join('.') const errorMessage = "Failed to retrieve script information or there are no available scripts for this domain." let neverLoadedScripts = true let collapsed = true let loadedPages = 0 function getScriptsInfo(domain, page = 1) { var url = `https://greasyfork.org/scripts/by-site/${domain}?filter_locale=0&sort=updated&page=${page}` GM_xmlhttpRequest({ method: "GET", url: url, onload: (response) => { // 解析结果 const parser = new DOMParser() const doc = parser.parseFromString(response.responseText, "text/html") const scripts = doc.querySelector("#browse-script-list")?.querySelectorAll('[data-script-id]') let scriptsInfo = [] if (!scripts) { scriptsInfo = errorMessage } else { for (var i = 0; i < scripts.length; i++) { scriptsInfo.push(parseScriptInfo(scripts[i])) } } // 处理对象 const loadMoreButton = document.querySelector('.load-more') console.log(doc.querySelector('.next_page')) if (doc.querySelector('.next_page') == null || doc.querySelector('.next_page')?.getAttribute('aria-disabled') === 'true') { loadedPages = 'max' loadMoreButton.disabled = true loadMoreButton.textContent = 'All scripts loaded' } else { loadMoreButton.disabled = false loadMoreButton.textContent = 'Load more' } // console.log(scriptsInfo); document.querySelector('.wait-loading').style.display = 'none' loadMoreButton.style.display = 'block' appendScriptsInfo(scriptsInfo) updateMatches() typeof (loadedPages) === 'number' ? loadedPages += 1 : loadedPages = loadedPages // console.log(loadedPages) }, onerror: () => { console.log("Some error occurred!") if (loadedPages === 0) { appendScriptsInfo(scriptsInfo) } const scriptsInfo = errorMessage document.querySelector('.wait-loading').style.display = 'none' } }) } // 解析脚本信息 function parseScriptInfo(script) { return { id: script.getAttribute('data-script-id'), name: script.getAttribute('data-script-name'), author: script.querySelector("dd.script-list-author").textContent, description: script.querySelector(".script-description").textContent, version: script.getAttribute('data-script-version'), url: 'https://greasyfork.org/scripts/' + script.getAttribute('data-script-id'), createDate: script.getAttribute('data-script-created-date'), updateDate: script.getAttribute('data-script-updated-date'), installs: script.getAttribute('data-script-total-installs'), dailyInstalls: script.getAttribute('data-script-daily-installs'), ratingScore: script.getAttribute('data-script-rating-score') } } // 插入脚本 function appendScriptsInfo(scriptsInfo) { const infoList = document.querySelector('.info-list') if (scriptsInfo === errorMessage) { // infoList.innerHTML = errorMessage; const loadMoreButton = document.querySelector('.load-more') loadMoreButton.disabled = true loadMoreButton.textContent = 'All scripts loaded' loadMoreButton.innerHTML = errorMessage } else { for (var i = 0; i < scriptsInfo.length; i++) { var script = scriptsInfo[i] var listItem = document.createElement('li') listItem.className = 'info-item' var scriptContainer = document.createElement('div') scriptContainer.className = 'script-container' var nameElement = document.createElement('a') nameElement.className = 'mscript-link' nameElement.innerText = script.name nameElement.href = script.url nameElement.target = '_blank' var descriptionElement = document.createElement('p') descriptionElement.className = 'script-description' descriptionElement.innerHTML = script.description var detailsContainer = document.createElement('div') detailsContainer.className = 'details-container' // 创建一键安装按钮 var installButton = document.createElement('a') installButton.className = 'install-button' installButton.innerText = `Install ${script.version}` installButton.href = `https://greasyfork.org/scripts/${script.id}/code/script.user.js` const details = [ { key: 'Author', value: script.author }, { key: 'Installs', value: script.installs }, { key: 'Daily Installs', value: script.dailyInstalls }, { key: 'Created', value: script.createDate }, { key: 'Updated', value: script.updateDate }, { key: 'Rating', value: script.ratingScore } ] for (let i = 0; i < details.length; i++) { const spanElement = document.createElement('span') spanElement.className = 'script-details' spanElement.innerText = `${details[i].key}:\n${details[i].value}` detailsContainer.appendChild(spanElement) } scriptContainer.appendChild(nameElement) scriptContainer.appendChild(descriptionElement) scriptContainer.appendChild(detailsContainer) scriptContainer.appendChild(installButton) listItem.appendChild(scriptContainer) listItem.scriptId = script.id infoList.appendChild(listItem) } } } function setupUI() { GM_addStyle(` scrbutton.script-button { position: fixed; bottom: 20%; right: -50px; transform: translateY(50%); padding: 20px; font-size: 16px; border: none; border-radius: 4px; background-color: #1e90ff; color: #ffffff; cursor: pointer; transition: right 0.3s; z-index: 9999999999999999; } div.info-container { display: none; position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); width: 650px; padding: 12px; background-color: #ffffff; box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3); border-radius: 4px; opacity: 0; transition: opacity 0.3s; z-index: 9999; max-height: 80vh; overflow-y: auto; } ul.info-list { list-style: none; margin: 0; padding: 0; } li.info-item { margin-bottom: 15px; padding: 12px; padding-bottom: 22px; display: flex; flex-direction: column; border: 1px solid #1e90ff; border-radius: 5px; } .div.script-container { display: flex; flex-direction: column; } a.mscript-link { font-size: 18px !important; font-weight: bold !important; margin-bottom: 5px !important; color: #1e90ff !important; } p.script-description { color: black !important; margin-top: 2px; margin-bottom: 5px; font-size: 16px; } div.details-container { font-size: 15px; font-weight: bold; display: flex; justify-content: space-between; margin-bottom: 15px; } span.script-details { font: !important; color: black !important; flex-grow: 1 !important; text-align: center !important; border: 1px solid #1e90ff !important; border-radius: 5px !important; margin: 4px !important; } div.table-header { color: #1e90ff !important; font-size: 25px; font-weight: bold; } input.script-search-input { width: 96% !important; padding: 10px !important; font-size: 18px !important; border: 1px solid #1e90ff !important; border-radius: 4px !important; box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3) !important; margin-bottom: 15px !important; margin-top: 20px !important; } a.install-button { font-size: 20px; background-color: green; color: white; padding: 12px; } button.to-greasyfork { position: absolute; top: 12px; right: 12px; border-radius: 4px; padding: 8px; font-size: 16px; border: none; background-color: #1e90ff; color: #ffffff; cursor: pointer; } span.match-count { background-color: #1e90ff; font-size: 25px; font-weight: bold; color: white; padding: 6px; position: absolute; right: 50%; border-radius: 12px; top: 10px; } div.wait-loading { font-size: 20px; font-weight: bold; color: #1e90ff; animation: blink 1s infinite; } @keyframes fadeInOut { 0% { opacity: 0; } 50% { opacity: 1; } 100% { opacity: 0; } } @keyframes blink { 0%, 100% { opacity: 0; } 50% { opacity: 1; } } button.load-more { border-radius: 4px; padding: 8px; font-size: 16px; border: none; background-color: #1e90ff; color: #ffffff; cursor: pointer; position: relative; bottom: 5px; left: 50%; transform: translateX(-50%); } button.load-more:disabled { background-color: #cccccc; cursor: not-allowed; } /* Mobile styles */ @media (max-width: 600px) { scrbutton.script-button { right: -30px; padding: 8px; font-size: 14px; } span.script-details { font-size: 10px !important; margin: 2px !important; padding: 2px !important; } span.match-count { font-size: 20px; padding: 4px; } button.to-greasyfork { padding: 6px; font-size: 14px; } a.install-button { font-size: 12px; padding: 8px; } div.table-header { font-size: 20px; } div.script-container { padding: 10px; } div.info-container { top: 10%; left: 5%; right: 5%; transform: none; width: calc(90% - 10px); /* 自适应宽度,保持左右边距 */ max-width: 100%; /* 确保不超出屏幕宽度 */ } a.script-link { font-size: 16px !important; } p.script-description { font-size: 14px; } { input.script-search-input { width: 92% !important; padding: 8px !important; font-size: 16px !important; } span.match-count { font-size: 20px; padding: 4px; } button.load-more { font-size: 14px; padding: 6px; } } `) // 创建打开列表按钮 var button = document.createElement('scrbutton') button.className = 'script-button' button.innerText = 'Scripts' // 创建脚本容器 var infoContainer = document.createElement('div') infoContainer.className = 'info-container' // 创建搜索框 var searchInput = document.createElement('input') searchInput.type = 'text' searchInput.placeholder = 'Search scripts...' searchInput.className = 'script-search-input' // 创建指向greasyfork的链接 var toGreasyfork = document.createElement('button') toGreasyfork.className = 'to-greasyfork' toGreasyfork.innerText = 'View on Greasyfork' // 创建计数器 var matchCount = document.createElement('span') matchCount.className = 'match-count' // 创建表头 var tableHeader = document.createElement('div') tableHeader.className = 'table-header' tableHeader.appendChild(document.createTextNode('Script Finder')) tableHeader.appendChild(matchCount) tableHeader.appendChild(searchInput) tableHeader.appendChild(toGreasyfork) // 创建脚本列表 var infoList = document.createElement('ul') infoList.className = 'info-list' // 创建等待加载 var waitLoading = document.createElement('div') waitLoading.className = 'wait-loading' waitLoading.innerText = 'Loading scripts...' // 创建加载更多 var loadMore = document.createElement('button') loadMore.className = 'load-more' loadMore.innerText = 'Load more' loadMore.style.display = 'none' infoList.appendChild(waitLoading) infoList.appendChild(loadMore) infoContainer.appendChild(tableHeader) infoContainer.appendChild(infoList) var timeout button.addEventListener('mouseenter', function () { clearTimeout(timeout) button.style.right = '10px' }) button.addEventListener('mouseleave', function () { timeout = setTimeout(function () { button.style.right = '-50px' }, 500) }) button.addEventListener('click', function (event) { event.stopPropagation() if (collapsed) { infoContainer.style.display = "block" infoContainer.style.opacity = 1 collapsed = false } else { infoContainer.style.display = "none" infoContainer.style.opacity = 0 collapsed = true } if (neverLoadedScripts) { getScriptsInfo(domain, 1) neverLoadedScripts = false } }) infoContainer.addEventListener('click', function (event) { event.stopPropagation() }) searchInput.addEventListener('input', () => { searchScript() updateMatches() }) toGreasyfork.addEventListener('click', function () { window.open(`https://greasyfork.org/scripts/by-site/${domain}?q=${searchInput.value}&filter_locale=0&sort=updated`) }) loadMore.addEventListener('click', () => { if (loadedPages === 'max') { return } const loadMoreButton = document.querySelector('.load-more') loadMoreButton.disabled = true loadMoreButton.textContent = 'Loading...' document.querySelector('.wait-loading').style.display = 'block' getScriptsInfo(domain, loadedPages + 1) }) document.body.addEventListener('click', function () { clearTimeout(timeout) collapsed = true button.style.right = '-50px' infoContainer.style.opacity = 0 infoContainer.style.display = "none" }) document.body.appendChild(button) document.body.appendChild(infoContainer) infoContainer.addEventListener('change', () => { updateMatches() }) updateMatches() } function searchScript() { const searchWord = document.querySelector('.script-search-input').value.toLowerCase() // 将要匹配的文本转换为小写 const scriptList = document.querySelectorAll('.info-item') for (let i = 0; i < scriptList.length; i++) { const scriptText = scriptList[i].innerText.toLowerCase() // 将检索的文本转换为小写 if (scriptText.includes(searchWord)) { scriptList[i].style.display = 'block' } else { scriptList[i].style.display = 'none' } } } function updateMatches() { const matchCount = document.querySelectorAll('.info-item:not([style*="display: none"])').length const allCount = document.querySelectorAll('.info-item').length document.querySelector('.match-count').innerText = matchCount === allCount ? matchCount : `${matchCount}/${allCount}` } function main() { if (window.self !== window.top) { // 在iframe中执行时,直接退出 return } setupUI() } main() })()