一个轻量级的搜索引擎优化脚本。自动净化百度、谷歌、必应等搜索结果页面,支持移除广告、优化重定向链接、清理URL、隐藏弹窗等功能,让搜索体验更简洁高效
// ==UserScript== // @name 搜索引擎净化 // @namespace https://github.com/QingJ01/Search_clear // @version 1.1.0 // @icon  // @description 一个轻量级的搜索引擎优化脚本。自动净化百度、谷歌、必应等搜索结果页面,支持移除广告、优化重定向链接、清理URL、隐藏弹窗等功能,让搜索体验更简洁高效 // @author QingJ // @license Apache Licence 2.0 // @match *://*.google.com/* // @match *://*.google.com.hk/* // @match *://*.google.co.jp/* // @match *://*.baidu.com/* // @match *://*.bing.com/* // @match *://*.cn.bing.com/* // @match *://*.s.cn.bing.net/* // @match *://*.sogou.com/* // @match *://*.m.sogou.com/* // @match *://*.wap.sogou.com/* // @match *://*.so.com/* // @match *://*.m.so.com/* // @match *://*.sm.cn/* // @match *://*.m.sm.cn/* // @match *://*.yz.m.sm.cn/* // @match *://*.so.toutiao.com/* // @grant GM_addStyle // @grant unsafeWindow // @grant GM_getValue // @grant GM_setValue // @grant GM_openInTab // @grant GM_xmlhttpRequest // @connect api.staticj.top // @run-at document-start // @require https://code.jquery.com/jquery-3.6.0.min.js // ==/UserScript== (function (cat) { "use strict"; const userConfig = { css: " {display: none !important;width: 0 !important;height: 0 !important;} ", timeout: 10000, tryCount: 5, tryTimeout: 500, }; const commonFunctionObject = { GMgetValue(key, defaultValue) { return GM_getValue(key, defaultValue); }, GMsetValue(key, value) { GM_setValue(key, value); }, GMopenInTab(url) { GM_openInTab(url, { active: true }); }, randomNumber() { return Math.floor(Math.random() * 100000000); }, webToast(config) { const message = config.message || ''; const background = config.background || '#333'; const toast = document.createElement('div'); toast.style.cssText = ` position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); padding: 10px 20px; background: ${background}; color: #fff; border-radius: 4px; z-index: 999999; `; toast.textContent = message; document.body.appendChild(toast); setTimeout(() => toast.remove(), 2000); }, request(method, url, data) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: method, url: url, data: data, onload: function (response) { resolve(response); }, onerror: function (error) { reject(error); } }); }); } }; /** * 常量定义 */ const HOSTS = { BAIDU: 'baidu.com', GOOGLE: 'google.com', BING: 'bing.com', SOGOU: 'sogou.com', SO: 'so.com', SM: 'sm.cn' }; /** * 工具函数:移除元素 */ function removeElements(selector) { try { document.querySelectorAll(selector).forEach(el => el.remove()); } catch (e) { console.error('移除元素失败:', selector, e); } } /** * 检查当前域名是否匹配 */ function isMatchHost(host) { return location.host.includes(host); } /** * 添加样式 */ function addStyle(css, pass = 0) { let el; if (pass >= userConfig.tryCount) return; if (typeof cat.GM_addStyle == "function") { el = cat.GM_addStyle(css); } else { el = document.createElement("style"); el.textContent = css; document.documentElement.appendChild(el); } if (typeof el == "object") { if (!el || !document.documentElement.contains(el)) { setTimeout(() => { addStyle(css, pass + 1); }, userConfig.tryTimeout); } } } /** * 搜索引擎广告过滤 */ function removeAds() { // 百度广告过滤 if (isMatchHost(HOSTS.BAIDU)) { // 移除顶部和右侧广告 removeElements('.ec_wise_ad'); removeElements('#content_right'); // 移除带有广告标记的内容 - 更精确的选择器 document.querySelectorAll('#content_left > div').forEach(container => { // 检查是否包含广告标记 if (container.querySelector('.f13 > span')?.textContent?.includes('广告') || container.querySelector('a[data-landurl]') || container.querySelector('span.tuiguang')?.textContent?.includes('广告') || container.querySelector('span.brand')?.textContent?.includes('广告') || container.querySelector('span[data-tuiguang]')) { container.remove(); } }); // 移除多余元素 - 更精确的选择器 removeElements('#content_right > br'); removeElements('#content_right > div:not([id])'); removeElements('#content_left > div.result-op'); removeElements('#content_left > div[class*="recommend"]'); // 移除劫持和推荐 removeElements('.res_top_banner'); removeElements('#content_left div[class*="_rs"]'); // 移除手机版广告 if (location.host.includes('m.baidu.com')) { // 基础广告元素 removeElements([ '.ec_wise_ad', '.ec-result-inner', '.c-result.result-op', '.download-tip', '.float-ball', '.ball-wrapper', '.na-like-container' ].join(',')); // 移除带有广告标记的内容 document.querySelectorAll('.c-container').forEach(container => { if (container.querySelector('.c-icons-outer')?.textContent?.includes('广告') || container.hasAttribute('data-tuiguang') || container.querySelector('[data-tuiguang]')) { container.remove(); } }); // 移除底部推广 removeElements([ '.c-recommends', '.c-flex-recommend', '.c-recommend-tip', '[data-module="recommend"]' ].join(',')); } } // 谷歌广告过滤 if (isMatchHost(HOSTS.GOOGLE)) { removeElements([ '.commercial-unit', '#tads', '#bottomads', 'div[aria-label="广告"]', 'div[aria-label="Ads"]' ].join(',')); } // 必应广告过滤 if (isMatchHost(HOSTS.BING)) { removeElements([ 'li.b_ad', '.pa_sb', '.adsMvC', 'a[h$=",Ads"]', 'a[href*="/aclick?ld="]', 'DIV#bnp_container', '.ad_sc' ].join(',')); // 移除特定图片的广告 document.querySelectorAll('.b_algo').forEach(algo => { const img = algo.querySelector('.rms_img'); if (img && (img.src.includes('/th?id=OADD2.') || img.src.includes('=AdsPlus'))) { algo.remove(); } }); } // 搜狗广告过滤 if (isMatchHost(HOSTS.SOGOU)) { removeElements([ '#so_kw-ad', '#m-spread-left', '#m-spread-bottom' ].join(',')); // 移除带有广告标记的内容 document.querySelectorAll('#righttop_box li').forEach(li => { if (li.querySelector('span')?.textContent.includes('广告')) { li.remove(); } }); } // 360搜索广告过滤 if (isMatchHost(HOSTS.SO)) { removeElements([ '.res-mediav', '.e_result', '.c-title-tag', 'DIV.res-mediav-right', 'DIV.inner_left', '#so-activity-entry', 'DIV.tg-wrap' ].join(',')); } // 神马搜索广告过滤 if (isMatchHost(HOSTS.SM)) { removeElements([ '.ad-wrapper', '.ec_wise_ad', '.qb-download-banner-non-share', 'DIV[data-text-ad]', '.ad-block' ].join(',')); // 移除手机版广告 if (location.host.includes('m.sm.cn')) { removeElements([ 'DIV.ad-alert-info', '.se-recommend-word-list-container', '#se-recommend-word-list-container', '[class*="ball-wrapper"]', 'DIV#page-copyright.se-page-copyright[style*="margin-bottom: 50px"]', 'DIV[style*="position: fixed; bottom: 0px"]', '[ad_dot_url*="http"]', '.dl-banner-without-logo', '.ad_result', '.biz_sponsor' ].join(',')); } } } /** * 搜索引擎重定向优化 */ function redirectOptimize() { // 处理百度重定向 if (isMatchHost(HOSTS.BAIDU)) { document.querySelectorAll('a[href*="baidu.com/link"]:not([ac-redirect-processed])').forEach(link => { try { // 标记已处理,避免重复 link.setAttribute('ac-redirect-processed', '1'); // 保存原始链接 const originalHref = link.href; // 移除原有的点击事件 link.removeAttribute('onclick'); link.removeAttribute('onmousedown'); // 使用GM_xmlhttpRequest处理重定向 link.addEventListener('click', function (e) { e.preventDefault(); e.stopPropagation(); GM_xmlhttpRequest({ method: "GET", url: originalHref, headers: { "Accept": "*/*", "Referer": originalHref }, timeout: 5000, onload: function (response) { let directUrl = response.finalUrl; if (!directUrl) { // 尝试从响应文本中提取URL const matches = /URL='([^']+)'/.exec(response.responseText); directUrl = matches ? matches[1] : null; } if (directUrl && !directUrl.includes('baidu.com/link')) { window.open(directUrl, '_blank'); } else { window.open(originalHref, '_blank'); } }, onerror: function () { window.open(originalHref, '_blank'); } }); }, { once: true }); // 确保事件只触发一次 } catch (e) { console.error('百度重定向处理失败:', e); } }); } // 处理谷歌重定向 if (isMatchHost(HOSTS.GOOGLE)) { document.querySelectorAll('a[onmousedown]:not([ac-redirect-processed]), a[data-jsarwt]:not([ac-redirect-processed])').forEach(link => { try { // 标记已处理 link.setAttribute('ac-redirect-processed', '1'); // 移除Google的重定向属性 link.removeAttribute('onmousedown'); link.removeAttribute('data-jsarwt'); link.removeAttribute('ping'); // 新标签页打开 link.setAttribute('target', '_blank'); } catch (e) { console.error('谷歌重定向处理失败:', e); } }); } // 处理必应重定向 if (isMatchHost(HOSTS.BING)) { document.querySelectorAll('a[href*="/click?"]:not([ac-redirect-processed]), a[href*="go.microsoft.com"]:not([ac-redirect-processed])').forEach(link => { try { // 标记已处理 link.setAttribute('ac-redirect-processed', '1'); // 尝试从href中提取目标URL const url = new URL(link.href); let targetUrl = url.searchParams.get('u') || url.searchParams.get('r'); if (targetUrl) { // 设置解码后的URL link.href = decodeURIComponent(targetUrl); link.setAttribute('target', '_blank'); // 移除原有的点击事件 link.removeAttribute('onclick'); link.removeAttribute('onmousedown'); } } catch (e) { console.error('必应重定向处理失败:', e); } }); } } /** * 隐藏必应APP弹窗 */ function hideBingPopup() { if (isMatchHost(HOSTS.BING)) { addStyle('div#bnp_container {display: none !important;}'); const observer = new MutationObserver(() => { const closeBtn = document.querySelector('div#sacs_close'); if (closeBtn) { closeBtn.click(); } }); observer.observe(document.body, { childList: true, subtree: true }); } } /** * URL参数清理 */ function shortenUrl() { sturl(); window.addEventListener('locationchange', sturl); } function sturl() { try { let url = new URL(window.location.href); let changed = false; // 需要移除的查询参数 const params = [ // 百度参数 - 完整列表 'rsp', 'prefixsug', 'fr', 'bsst', 'f', 'inputT', 'usm', 'rsv_page', 'rqlang', 'rsv_t', 'oq', 'rsv_pq', 'rsv_spt', 'ie', 'rsv_enter', 'rsv_sug1', 'rsv_sug7', 'rsv_sug2', 'rsv_sug3', 'rsv_iqid', 'rsv_bp', 'rsv_btype', 'rsv_idx', 'rsv_dl', 'issp', 'cshid', 'tn', 'rsv_sug4', 'rtt', 'bsst', 'rsv_sid', 'rsv_tn', '_ss', 'rsv_jmp', 'rsv_bl', 'rsv_sug5', 'rsv_sug6', 'rsv_sug8', 'rsv_sug9', 'bar', // 谷歌参数 - 完整列表 'tbas', 'ved', 'uact', 'ei', 'ie', 'oq', 'sclient', 'cshid', 'dpr', 'iflsig', 'aqs', 'gs_lcp', 'source', 'sourceid', 'sxsrf', 'pccc', 'sa', 'biw', 'bih', 'hl', 'newwindow', 'stick', 'gws_rd', 'client', 'gs_rn', 'tbo', 'dcr', 'safe', 'ssui', 'psi', // 必应参数 - 完整列表 'tsc', 'sp', 'FORM', 'form', 'pq', 'sc', 'qs', 'sk', 'cvid', 'lq', 'ghsh', 'ghacc', 'ghpl', 'ghc', 'ubireng', 'FPIG', 'PC', 'setmkt', 'setlang', 'mkt', 'qpvt', 'ensearch', 'first' ]; // 移除参数 params.forEach(param => { if (url.searchParams.has(param)) { url.searchParams.delete(param); changed = true; } }); // 特殊参数处理 const specialParams = [ ['start', '0'], ['page', '1'], ['offset', '0'], ['first', '1'] ]; if (url.searchParams.get('start') === '0') { url.searchParams.delete('start'); changed = true; } // 如果有改变则更新URL if (changed) { window.history.replaceState(null, null, url.toString()); } } catch (e) { console.error('URL参数清理失败:', e); } } // 添加 locationchange 事件支持 const originalPushState = history.pushState; const originalReplaceState = history.replaceState; history.pushState = function () { originalPushState.apply(this, arguments); window.dispatchEvent(new Event('locationchange')); }; history.replaceState = function () { originalReplaceState.apply(this, arguments); window.dispatchEvent(new Event('locationchange')); }; window.addEventListener('popstate', () => { window.dispatchEvent(new Event('locationchange')); }); // SearchEnginesNavigation 类定义 class SearchEnginesNavigation { constructor() { this.navigationDataCache = "navigation_data_cache"; this.customNavigationkey = "custom-navigation-key-8898"; this.searchEnginesData = [ { "host": "www.baidu.com", "element": "#content_right", "elementInput": "#kw" }, { "host": "www.so.com", "element": "#side", "elementInput": "#keyword" }, { "host": "www.sogou.com", "element": "#right", "elementInput": "#upquery" }, { "host": "cn.bing.com", "element": "#b_context", "elementInput": "#sb_form_q" }, { "host": "www.bing.com", "element": "#b_context", "elementInput": "#sb_form_q" }, { "host": "www4.bing.com", "element": "#b_context", "elementInput": "#sb_form_q" }, { "host": "so.toutiao.com", "element": ".s-side-list", "elementInput": "input[type='search']" }, { "host": "www.google.com", "element": "#rhs", "elementInput": "input[name='q']" } ]; } getNavigationData(element, elementInput) { const defaultNavigationData = [ { "name": "搜索引擎", "list": [ { "name": "百度", "url": "https://www.baidu.com/s?wd=@@" }, { "name": "必应", "url": "https://cn.bing.com/search?q=@@" }, { "name": "Google", "url": "https://www.google.com/search?q=@@" }, { "name": "360搜索", "url": "https://www.so.com/s?ie=utf-8&fr=none&src=360sou_newhome&nlpv=basest&q=@@" }, { "name": "搜狗", "url": "https://www.sogou.com/web?query=@@" }, { "name": "头条搜索", "url": "https://so.toutiao.com/search?dvpf=pc&source=input&keyword=@@" } ] }, { "name": "资源搜索", "list": [ { "name": "财经搜索", "url": "https://www.shaduizi.com/s/search?q=@@¤tPage=1" }, { "name": "百度百科", "url": "https://baike.baidu.com/item/@@" }, { "name": "知乎搜索", "url": "https://www.zhihu.com/search?type=content&q=@@" }, { "name": "B站搜索", "url": "https://search.bilibili.com/all?keyword=@@&from_source=webtop_search&spm_id_from=333.851" }, { "name": "抖音搜索", "url": "https://www.douyin.com/search/@@?aid=0a9fc74b-01e8-4fb0-9509-307c5c07fda1&publish_time=0&sort_type=0&source=normal_search&type=general" }, { "name": "搜狗|公众号", "url": "https://weixin.sogou.com/weixin?type=2&query=@@" }, { "name": "搜狗|知乎", "url": "https://www.sogou.com/sogou?pid=sogou-wsse-ff111e4a5406ed40&insite=zhihu.com&ie=utf8&p=73351201&query=@@&ie=utf8&p=73351201&query=@@" }, { "name": "豆瓣搜索", "url": "https://www.douban.com/search?q=@@" }, { "name": "电影搜索", "url": "https://www.cupfox.com/search?key=@@" }, { "name": "维基百科", "url": "https://en.wikipedia.org/w/index.php?search=@@" }, { "name": "法律法规", "url": "https://www.pkulaw.com/law/chl?Keywords=@@" }, { "name": "icon搜索", "url": "https://www.iconfont.cn/search/index?searchType=icon&q=@@" }, { "name": "github", "url": "https://github.com/search?q=@@" }, { "name": "csdn", "url": "https://so.csdn.net/so/search?q=@@&t=&u=" }, { "name": "stackoverflow", "url": "https://stackoverflow.com/" } ] } ]; let cacheNavigationData = commonFunctionObject.GMgetValue(this.navigationDataCache, null); if (!cacheNavigationData) { cacheNavigationData = defaultNavigationData; } let finalNavigationData = null; try { let customNavigationData = commonFunctionObject.GMgetValue(this.customNavigationkey, null); finalNavigationData = customNavigationData ? cacheNavigationData.concat(customNavigationData) : cacheNavigationData; } catch (e) { finalNavigationData = cacheNavigationData; } this.createHtml(element, elementInput, finalNavigationData); // 更新缓存数据 commonFunctionObject.request("get", "http://api.staticj.top/script/api/get/navigation_json_url?t=" + new Date().getTime(), null) .then(resultData => { let dataJson = JSON.parse(resultData.data); if (dataJson?.url) { commonFunctionObject.request("get", dataJson.url, null) .then(resultData2 => { let serverNavigationData = resultData2.data; if (!cacheNavigationData || (cacheNavigationData && serverNavigationData.length != JSON.stringify(cacheNavigationData).length)) { commonFunctionObject.GMsetValue(this.navigationDataCache, JSON.parse(serverNavigationData)); } }) .catch(() => { }); } }) .catch(() => { }); } createCss(elementNum) { const cssContent = ` /* 导航容器样式 */ #QingjByeBug { background: #fff; border-radius: 8px; box-shadow: 0 2px 8px rgba(0,0,0,0.1); padding: 15px; margin-bottom: 20px; } /* 分类样式 */ .ddfdfd${elementNum}dffssqa { margin-top: 13px; } .ddfdfd${elementNum}dffssqa:first-child { margin-top: 0; } /* 标题样式 */ .ddfdfd${elementNum}dffssqa>.title { font-size: 15px; color: #333; margin-bottom: 10px; padding-bottom: 8px; border-bottom: 1px solid #eee; } .ddfdfd${elementNum}dffssqa>.title b { position: relative; padding-left: 10px; } .ddfdfd${elementNum}dffssqa>.title b:before { content: ''; position: absolute; left: 0; top: 50%; transform: translateY(-50%); width: 4px; height: 16px; background: #4e6ef2; border-radius: 2px; } /* 链接列表样式 */ .ddfdfd${elementNum}dffssqa>.content-list { display: flex; flex-wrap: wrap; gap: 8px; } /* 链接样式 */ .ddfdfd${elementNum}dffssqa>.content-list>a { flex: 0 0 calc(31% - 4px); text-decoration: none; color: #333; background: #f5f5f5; border: 1px solid transparent; border-radius: 4px; padding: 6px 2px; text-align: center; font-size: 13px; line-height: 1.5; overflow: hidden; white-space: nowrap; text-overflow: ellipsis; transition: all 0.3s ease; } .ddfdfd${elementNum}dffssqa>.content-list>a:hover { background: #4e6ef2; color: #fff; transform: translateY(-1px); box-shadow: 0 2px 8px rgba(78,110,242,0.3); } /* 底部信息样式 */ #QingjByeBug > div:last-child { margin-top: 15px; padding-top: 10px; border-top: 1px solid #eee; color: #999; font-size: 12px; } #QingjByeBug > div:last-child a { color: #666; text-decoration: none; transition: color 0.3s ease; } #QingjByeBug > div:last-child a:hover { color: #4e6ef2; } /* 自定义按钮样式 */ a[name="customNavigation"] { display: inline-flex; align-items: center; gap: 4px; } a[name="customNavigation"]:before { content: '🔧'; font-size: 14px; } `; if ($("#plugin_css_style_dddsoo").length == 0) { $("body").prepend(`<style id='plugin_css_style_dddsoo'>${cssContent}</style>`); } } createHtml(element, elementInput, navigationData) { $("#QingjByeBug").remove(); // 保持原有ID const elementNum = commonFunctionObject.randomNumber(); let isComplete = true; // 添加这一行,确保CSS样式被添加 this.createCss(elementNum); const elementInterval = setInterval(() => { if (isComplete) { const $element = $(element); const $box = $("#QingjByeBug"); isComplete = false; if ($element.length != 0 && $box.length == 0) { let html = "<div id='QingjByeBug'>"; // 遍历导航分类 navigationData.forEach(category => { html += ` <div class='ddfdfd${elementNum}dffssqa'> <div class='title'><b>${category.name}</b></div> <div class='content-list'> `; // 遍历分类下的链接 category.list.forEach(item => { html += `<a target='_blank' name='navigation' data-url='${item.url}' href='javascript:void(0);'>${item.name}</a>`; }); html += "</div></div>"; }); html += ` <div style='margin-bottom:10px;margin-top:5px;font-size:12px;'> <a target='_blank' href='https://greasyfork.org/zh-CN/scripts/520018'>*该数据由 搜索引擎净化 提供</a> <a href="javascript:void(0);" name="customNavigation">自定义网址</a> </div> </div>`; // 插入导航面板 $element.prepend(html); // 绑定链接点击事件 $("#QingjByeBug a[name='navigation']").on("click", function (e) { e.preventDefault(); const url = $(this).data("url").replace("@@", $(elementInput).val()); commonFunctionObject.GMopenInTab(url); }); // 绑定自定义导航事件 $("#QingjByeBug a[name='customNavigation']").on("click", function (e) { e.preventDefault(); self.showSetingDialog(); }); } isComplete = true; } }, 100); } showSetingDialog() { const customNavigation = commonFunctionObject.GMgetValue(this.customNavigationkey, null); const customNavigationData = customNavigation ? JSON.stringify(customNavigation, null, 4) : ""; const content = ` <div class="custom-navigation-dialog"> <div class="notice-section"> <h3>注意事项:</h3> <ol> <li>请严格按照格式添加,否则不生效</li> <li>数据为json格式,请确保json格式正确,必要时请到 <a target="_blank" href="https://www.json.cn/">json.cn</a> 校验</li> <li>点击下面"示例"按钮,查看具体格式情况</li> <li>链接中的搜索关键词请用"@@"代替,脚本会自动替换成当前搜索词</li> <li>清空 -> 保存,则取消自定义的导航网址</li> </ol> </div> <div class="textarea-section"> <textarea placeholder="请严格按照格式填写,否则不生效" class="navigation-textarea" >${customNavigationData}</textarea> </div> <div class="button-section"> <button class="navigation-example">示例</button> <button class="navigation-clear">清空</button> <button class="navigation-save">保存自定义导航</button> </div> </div> `; // 创建弹窗样式 const style = document.createElement('style'); style.textContent = ` .custom-navigation-dialog { font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif; color: #333; padding: 20px; } .custom-navigation-dialog .notice-section { background: #f8f9fa; border-radius: 8px; padding: 15px 20px; margin-bottom: 20px; } .custom-navigation-dialog .notice-section h3 { color: #e74c3c; margin: 0 0 10px 0; font-size: 16px; font-weight: 600; } .custom-navigation-dialog .notice-section ol { margin: 0; padding-left: 20px; } .custom-navigation-dialog .notice-section li { line-height: 1.6; margin-bottom: 5px; color: #666; font-size: 14px; } .custom-navigation-dialog .notice-section a { color: #2196f3; text-decoration: none; transition: color 0.3s; } .custom-navigation-dialog .notice-section a:hover { color: #1976d2; text-decoration: underline; } .custom-navigation-dialog .textarea-section { margin: 20px 0; } .custom-navigation-dialog .navigation-textarea { width: 100%; height: 200px; padding: 12px; border: 1px solid #ddd; border-radius: 8px; font-family: Consolas, Monaco, 'Courier New', monospace; font-size: 14px; line-height: 1.5; resize: none; background-color: #fff; transition: border-color 0.3s, box-shadow 0.3s; } .custom-navigation-dialog .navigation-textarea:focus { outline: none; border-color: #4e6ef2; box-shadow: 0 0 0 3px rgba(78,110,242,0.1); } .custom-navigation-dialog .button-section { text-align: center; margin-top: 20px; } .custom-navigation-dialog button { padding: 8px 20px; margin: 0 8px; border: none; border-radius: 6px; font-size: 14px; font-weight: 500; cursor: pointer; transition: all 0.3s ease; } .custom-navigation-dialog button:hover { transform: translateY(-1px); } .custom-navigation-dialog .navigation-example { background: #f5f5f5; color: #333; } .custom-navigation-dialog .navigation-example:hover { background: #e0e0e0; } .custom-navigation-dialog .navigation-clear { background: #ff4d4f; color: white; } .custom-navigation-dialog .navigation-clear:hover { background: #ff7875; } .custom-navigation-dialog .navigation-save { background: #4e6ef2; color: white; } .custom-navigation-dialog .navigation-save:hover { background: #6c87f5; } .custom-navigation-dialog-overlay { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0, 0, 0, 0.5); z-index: 999998; animation: fadeIn 0.3s ease; } .custom-navigation-dialog-container { position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); background: white; border-radius: 12px; box-shadow: 0 4px 24px rgba(0, 0, 0, 0.15); z-index: 999999; width: 90%; max-width: 600px; max-height: 90vh; overflow: auto; animation: slideIn 0.3s ease; } .custom-navigation-dialog-close { position: absolute; right: 16px; top: 16px; width: 28px; height: 28px; border-radius: 50%; border: none; background: #f5f5f5; color: #666; font-size: 20px; line-height: 28px; text-align: center; cursor: pointer; transition: all 0.3s ease; padding: 0; display: flex; align-items: center; justify-content: center; } .custom-navigation-dialog-close:hover { background: #ff4d4f; color: white; } @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } } @keyframes slideIn { from { opacity: 0; transform: translate(-50%, -48%); } to { opacity: 1; transform: translate(-50%, -50%); } } `; // 创建遮罩层 const overlay = document.createElement('div'); overlay.className = 'custom-navigation-dialog-overlay'; // 创建弹窗容器 const dialogContainer = document.createElement('div'); dialogContainer.className = 'custom-navigation-dialog-container'; dialogContainer.innerHTML = content; // 创建关闭按钮 const closeBtn = document.createElement('button'); closeBtn.className = 'custom-navigation-dialog-close'; closeBtn.innerHTML = '×'; dialogContainer.appendChild(closeBtn); // 添加样式和元素到页面 document.head.appendChild(style); document.body.appendChild(overlay); document.body.appendChild(dialogContainer); // 绑定事件 const textarea = dialogContainer.querySelector('.navigation-textarea'); const exampleJson = [{ "name": "我的导航", "list": [ { "name": "百度", "url": "https://www.baidu.com/s?wd=@@" }, { "name": "必应", "url": "https://cn.bing.com/search?q=@@" } ] }]; // 示例按钮事件 dialogContainer.querySelector('.navigation-example').onclick = () => { textarea.value = JSON.stringify(exampleJson, null, 4); }; // 清空按钮事件 dialogContainer.querySelector('.navigation-clear').onclick = () => { textarea.value = ''; }; // 保存按钮事件 dialogContainer.querySelector('.navigation-save').onclick = () => { const content = textarea.value; if (!content) { commonFunctionObject.GMsetValue(this.customNavigationkey, null); commonFunctionObject.webToast({ "message": "保存成功:数据为空", "background": "#4e6ef2" }); closeDialog(); return; } try { const contentJson = JSON.parse(content); if (this.validateNavigationFormat(contentJson)) { commonFunctionObject.GMsetValue(this.customNavigationkey, contentJson); commonFunctionObject.webToast({ "message": "保存成功", "background": "#4e6ef2" }); closeDialog(); // 刷新导航显示 this.show(); } else { commonFunctionObject.webToast({ "message": "格式错误,请更正", "background": "#ff4d4f" }); } } catch (e) { commonFunctionObject.webToast({ "message": "格式错误,请更正", "background": "#ff4d4f" }); } }; // 关闭弹窗函数 const closeDialog = () => { overlay.remove(); dialogContainer.remove(); style.remove(); }; // 关闭按钮事件 closeBtn.onclick = closeDialog; // 点击遮罩层关闭 overlay.onclick = closeDialog; // 阻止弹窗点击事件冒泡到遮罩 dialogContainer.onclick = (e) => e.stopPropagation(); } validateNavigationFormat(data) { if (!Array.isArray(data)) return false; return data.every(category => { if (typeof category !== 'object' || !category.name || !Array.isArray(category.list)) { return false; } return category.list.every(item => { return typeof item === 'object' && typeof item.name === 'string' && typeof item.url === 'string'; }); }); } show() { const host = window.location.host; const href = window.location.href; if ((host === "www.baidu.com") || (host === "www.so.com" && href.includes("www.so.com/s")) || (host === "www.sogou.com" && (href.includes("www.sogou.com/web") || href.includes("www.sogou.com/sogou"))) || (host === "cn.bing.com" && href.includes("cn.bing.com/search")) || (host === "www.bing.com" && href.includes("www.bing.com/search")) || (host === "www4.bing.com" && href.includes("www4.bing.com/search")) || (host === "so.toutiao.com" && href.includes("so.toutiao.com/search")) || (host === "www.google.com" && href.includes("www.google.com/search"))) { const currentSearchEnginesData = this.searchEnginesData.find(item => host === item.host); if (currentSearchEnginesData) { this.getNavigationData(currentSearchEnginesData.element, currentSearchEnginesData.elementInput); } } } start() { this.show(); } } /** * 初始化函数 */ function init() { removeAds(); redirectOptimize(); hideBingPopup(); shortenUrl(); new SearchEnginesNavigation().start(); } // 使用 MutationObserver 监听DOM变化 const observer = new MutationObserver(() => { removeAds(); redirectOptimize(); }); observer.observe(document, { childList: true, subtree: true, attributes: true, attributeFilter: ["id", "class"] }); // 初始化执行 if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', init); } else { init(); } })({ GM_info: typeof GM_info == "object" ? GM_info : {}, unsafeWindow: typeof unsafeWindow == "object" ? unsafeWindow : window, GM_addStyle: typeof GM_addStyle == "function" ? GM_addStyle : undefined, });