- // ==UserScript==
- // @name Script Finder
- // @name:zh-CN Script Finder 油猴脚本查找
- // @namespace http://tampermonkey.net/
- // @version 0.1.6
- // @description Script Finder allows you to find userscripts from greasyfork on any website.
- // @description:zh-CN Script Finder 在任何网站上找到适用于该网站的greasyfork油猴脚本
- // @author shiquda
- // @namespace https://github.com/shiquda/shiquda_UserScript
- // @supportURL https://github.com/shiquda/shiquda_UserScript/issues
- // @match *://*/*
- // @connect greatest.deepsurf.us
- // @icon 
- // @grant GM_xmlhttpRequest
- // @grant GM_addStyle
- // @license AGPL-3.0
- // ==/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://greatest.deepsurf.us/scripts/by-site/${domain}?filter_locale=0&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://greatest.deepsurf.us/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;
- } 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 = 'script-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://greatest.deepsurf.us/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(`
- button.script-button {
- position: fixed;
- bottom: 50%;
- right: -50px;
- transform: translateY(50%);
- padding: 10px;
- font-size: 16px;
- border: none;
- border-radius: 4px;
- background-color: #1e90ff;
- color: #ffffff;
- cursor: pointer;
- transition: right 0.3s;
- z-index: 9999999999999999; /* 设置一个较高的 z-index 值 */
- }
- div.info-container {
- display: none;
- position: fixed;
- top: 10%;
- right: 100px;
- 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.script-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;
- }
- `);
-
-
- // 创建打开列表按钮
- var button = document.createElement('button');
- 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://greatest.deepsurf.us/scripts/by-site/${domain}?q=${searchInput.value}&filter_locale=0`)
- })
-
- 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()
-
-
- })();