// ==UserScript== // @name 优化手机bing搜索 // @namespace f+bing // @version 2025.04.08 // @description bing搜索结果自动翻页+网址黑名单+自定义屏蔽关键词 // @icon  // @author f+ // @match https://cn.bing.com/search* // @grant GM_getValue // @grant GM_setValue // @grant GM_xmlhttpRequest // @grant GM_addStyle // @grant GM_registerMenuCommand // @grant window.onurlchange // @grant unsafeWindow // @run-at document-end // @connect // @noframes // ==/UserScript== (function () { "use strict"; GM_addStyle(` .banlist-block-btn{ border: 0px; border-radius: 0.3rem; background-color: #eee; color: #fff; padding: 0px; transition: 1.5s; opacity: 0.6; position: absolute; top: 0.5rem; right: 0.5rem; height: 2.5rem; width:2.5rem; font-size:1.5rem; font-weight:bold; z-index: 2; } .banlist-config-modal { animation: fadeInn 0.4s; position: fixed; top: 8%; left: 50%; transform: translateX(-50%); z-index: 9999; background: white; padding: 20px; box-shadow: 0 0 10px rgba(0,0,0,0.3); width: 84%; } .banlist-add-container { margin: 15px 0; color: #777; } .banlist-add-textarea { box-sizing:border-box; width: 100%; border: none; background:#eee; padding: 10px; height: 10rem; resize: vertical; color: #222; } .banlist-add-textarea::placeholder { color: #aaa; } .banlist-button-group { text-align: right; margin-top: 20px; } .banlist-action-btn { padding: 0.5rem 2rem; margin-left: 1rem; border: none; border-radius: 4px; cursor: pointer; background: #999; color: white; } .banlist-action-btn.confirm { background: #007bff; } .banlist-action-btn.cancel { background: #eee; color: #777; } .loader { width: 30%; height: 10px; margin-bottom: 2px; background: linear-gradient(#a6a698 0 0) left / 20px 10px no-repeat #e2e2e2; animation: l1 3s infinite linear; justify-self: center; transition: transform 0.2s; } .loader.done{ transform: scaleY(0); } /* 样式补丁 */ #b_results .b_tpcn .tilk{ background-color: unset !important; } @keyframes l1 { 50% {background-position: right } } @keyframes fadeInn { from { opacity: 0 ;transform: translate(-50%, 10%);} to { opacity: 1 ;transform: translate(-50%, 0);} } `); const banlistData = { _cache: { ["bingCite" /* GM_KEY.CITE */]: new Set(GM_getValue("bingCite" /* GM_KEY.CITE */, [])), ["global" /* GM_KEY.WORD */]: new Set(GM_getValue("global" /* GM_KEY.WORD */, [])), }, has(item, dataType) { if (!item || item.length === 0) return false; for (let blackWord of this._cache[dataType]) { if (item.includes(blackWord)) return true; } return false; }, add(item, dataType) { if (!item) return; this._cache[dataType].add(item); GM_setValue(dataType, [...this._cache[dataType]]); }, }; //-------------------------/ 创建配置窗口 /-------------------------// function createConfigWindow() { const modal = document.createElement("div"); modal.className = "banlist-config-modal"; // 添加屏蔽区域 const addSection = document.createElement("div"); addSection.className = "banlist-add-container"; const label = document.createElement("label"); label.textContent = "添加屏蔽词:"; const textArea = document.createElement("textarea"); textArea.className = "banlist-add-textarea"; textArea.placeholder = "每行输入一个屏蔽词"; addSection.append(label, textArea); // 操作按钮 const btnGroup = document.createElement("div"); btnGroup.className = "banlist-button-group"; const confirmBtn = document.createElement("banlist-button"); confirmBtn.className = `banlist-action-btn confirm`; confirmBtn.textContent = "保存"; confirmBtn.onclick = () => { saveConfig(textArea); modal.remove(); }; const cancelBtn = document.createElement("banlist-button"); cancelBtn.className = `banlist-action-btn cancel`; cancelBtn.textContent = "取消"; cancelBtn.onclick = () => modal.remove(); btnGroup.append(confirmBtn, cancelBtn); modal.append(addSection, btnGroup); document.body.appendChild(modal); } function saveConfig(textArea) { const newBlocked = textArea.value .split("\n") .map((v) => v.trim()) .filter((v) => v.length > 0); newBlocked.forEach((word) => banlistData.add(word, "global" /* GM_KEY.WORD */)); } //-------------------------/ 核心处理函数 /-------------------------// const processedNodes = new WeakSet(); // 存储已处理节点 // 检测是否加载完,同时处理屏蔽条目 function scanFisrtPage() { selectResults(document).forEach((ele) => { if (processedNodes.has(ele)) return; processedNodes.add(ele); if (isShouldBan(ele)) { removeElement(ele); } else { addBlockButton(ele); setNewTabOpen(ele); } }); } function parseNewResults(doc) { const goodResults = []; selectResults(doc).forEach((ele) => { if (processedNodes.has(ele)) return; processedNodes.add(ele); if (!isShouldBan(ele)) { addBlockButton(ele); transRmsImage(ele); setNewTabOpen(ele); goodResults.push(ele); } }); return goodResults; } function autoLoadPage() { let loadNextPage; let obPageReady; function initLoader() { // 确保每次初始化前断开已有监听 if (loadNextPage) window.removeEventListener("scroll", loadNextPage); if (obPageReady) obPageReady.disconnect(); loadNextPage = getNextPageLoader(); obPageReady = new MutationObserver(() => { const _isComplete = isFirstPageComplete(); // 需要先检测是否加载完再scan,因为scan会破坏检测条件 scanFisrtPage(); if (_isComplete) { loadNextPage(); window.addEventListener("scroll", loadNextPage); obPageReady.disconnect(); obPageReady = null; } }); obPageReady.observe(document.body, { childList: true, subtree: true, attributes: false, }); } // 拦截 pushState 和 replaceState 方法 if (window.onurlchange === undefined) { (function (history) { history.pushState = ((fn) => function () { const result = fn.apply(history, arguments); window.dispatchEvent(new Event("pushstate")); window.dispatchEvent(new Event("urlchange")); return result; })(history.pushState); history.replaceState = ((fn) => function () { const result = fn.apply(history, arguments); window.dispatchEvent(new Event("replacestate")); window.dispatchEvent(new Event("urlchange")); return result; })(history.replaceState); window.addEventListener("popstate", () => { window.dispatchEvent(new Event("urlchange")); }); })(window.history); } window.addEventListener("urlchange", () => { console.log(`URL changed: ${location.href}`); initLoader(); }); initLoader(); } function getNextPageLoader() { var _a, _b; let curUrl = location.href; let curDocument = document; let isLoading = false; const loadingElement = document.createElement("div"); loadingElement.className = "loader done"; (_b = (_a = document.body.querySelector("#b_results")) === null || _a === void 0 ? void 0 : _a.parentNode) === null || _b === void 0 ? void 0 : _b.append(loadingElement); function startLoad() { isLoading = true; loadingElement.classList.remove("done"); } function endLoad() { isLoading = false; loadingElement.classList.add("done"); } return async function loadNextPage() { var _a; if (isLoading) return; startLoad(); const isScrollBottom = (() => { const scrollTop = document.documentElement.scrollTop; const scrollHeight = document.body.scrollHeight; const clientHeight = document.documentElement.clientHeight; const isScrollBottom = scrollTop + clientHeight >= scrollHeight - 600; return isScrollBottom; })(); if (!isScrollBottom) { endLoad(); return; } try { const nextUrl = getNextUrl(curUrl, curDocument); const nextDocument = await getDocument(nextUrl, { anonymous: true }); const goodResults = parseNewResults(nextDocument); (_a = document.body.querySelector("#b_results")) === null || _a === void 0 ? void 0 : _a.append(...goodResults); // 处理新的结果 curUrl = nextUrl; curDocument = nextDocument; } catch (_b) { console.log("读取下一页失败!"); } endLoad(); loadNextPage(); }; } //-------------------------/ 工具函数 /-------------------------// // 提取搜索结果的条目 function selectResults(doc) { const selector = "#b_results> *:not(#progContPlaceHolder)"; return doc.querySelectorAll(selector); } // 解析条目的引用地址 function parseItemCite(ele) { const cite = ele.querySelector(".b_attribution cite"); if (cite) return cite.textContent; const link = ele.querySelector("a[href]"); if (link) return link.getAttribute("href"); return null; } // 判断条目 function isShouldBan(ele) { var _a, _b, _c, _d, _e; // 底部的通知 if (((_a = ele.classList) === null || _a === void 0 ? void 0 : _a.contains("b_msg")) && ((_b = ele.classList) === null || _b === void 0 ? void 0 : _b.contains("b_canvas"))) return true; // 相关搜索 if (ele.classList.contains("b_ans") && ((_c = ele.querySelector("h2")) === null || _c === void 0 ? void 0 : _c.textContent) == "相关搜索") return true; // 广告 if ((_d = ele.classList) === null || _d === void 0 ? void 0 : _d.contains("b_ad")) return true; // 下一页 if (ele.classList.contains("b_pag")) return true; // 黑名单标题 if (banlistData.has((_e = ele.querySelector("h2")) === null || _e === void 0 ? void 0 : _e.textContent, "global" /* GM_KEY.WORD */)) return true; // 黑名单链接 if (banlistData.has(parseItemCite(ele), "bingCite" /* GM_KEY.CITE */)) return true; return false; } // 为条目增加可以屏蔽该引用地址的按钮 function addBlockButton(ele) { let cite = parseItemCite(ele); if (!cite) return; try { const urlObj = new URL(cite); cite = urlObj.origin; } catch (e) { console.log("该条目的引用不是有效网址!"); } const btn = document.createElement("button"); btn.className = "banlist-block-btn"; btn.textContent = "临兵斗者皆阵列在前"[Date.now() % 9]; btn.addEventListener("click", (event) => { event.preventDefault(); event.stopImmediatePropagation(); // 彻底阻止传播 removeElement(ele); banlistData.add(cite, "bingCite" /* GM_KEY.CITE */); }); ele.appendChild(btn); } function setNewTabOpen(ele) { ele.querySelectorAll("a[href]").forEach((a) => { a.setAttribute("target", "_blank"); }); } // 清除黑名单条目的方法 function removeElement(ele) { ele.outerHTML = ""; } // bing搜索中网站图片采用懒加载,使其正常显示 function transRmsImage(ele) { // 将div.rms_iac元素改为图片 const rmsImages = ele.querySelectorAll("div.rms_iac[data-src]"); rmsImages.forEach((imgDiv) => { try { const img = document.createElement("img"); Object.keys(imgDiv.dataset).forEach((key) => { // 将 data-src 转换为 src,data-alt 转换为 alt,依此类推 const attrName = key.replace("data-", ""); img.setAttribute(attrName, imgDiv.dataset[key]); }); imgDiv.replaceWith(img); } catch (_a) { console.log("转换图片失败!"); } }); } // 判断第一页搜索结构是否已经显示完 function isFirstPageComplete() { // (document.readyState === "complete") return document.querySelector(".sb_fullnpl , .sb_halfnext") != null; } // bing默认每页加载10条,但是后5条则采用ajax加载。每页请求5条以内则可以绕过该问题。 const RESULT_PER_LOAD = 4; // 获得下一页地址,优先采用bing提供的下一页链接,其次采用人为构建 function getNextUrl(curUrl, curDocument) { var _a; const urlObj = (_a = _getNextUrlByNextButton(curUrl, curDocument)) !== null && _a !== void 0 ? _a : _getNextUrlBySearchParams(curUrl); urlObj.searchParams.set("count", `${RESULT_PER_LOAD}`); return urlObj.href; } function _getNextUrlByNextButton(curUrl, curDocument) { var _a; let nextUrl = (_a = curDocument.querySelector(".sb_fullnpl , .sb_halfnext")) === null || _a === void 0 ? void 0 : _a.getAttribute("href"); if (!nextUrl) return null; const urlObj = new URL(nextUrl, curUrl); let first = urlObj.searchParams.get("first"); if (first == null) return null; return urlObj; } function _getNextUrlBySearchParams(curUrl) { const urlObj = new URL(curUrl); let first = urlObj.searchParams.get("first"); if (first == null) { first = "8"; } else { first = String(parseInt(first) + RESULT_PER_LOAD); } urlObj.searchParams.set("first", first); return urlObj; } function getDocument(targetUrl, requestData = {}) { return new Promise((resolve, reject) => { const _requestData = { method: "GET", url: targetUrl, timeout: 5000, ...requestData, onload: function (response) { console.log(`获取内容成功:\n${targetUrl}`); const htmlContent = response.responseText; const parser = new DOMParser(); const doc = parser.parseFromString(htmlContent, "text/html"); resolve(doc); }, onerror: function (error) { console.log(`获取内容失败:\n${targetUrl}`); reject(error); }, ontimeout: function () { console.log(`获取内容超时:\n${targetUrl}`); reject(); }, }; GM_xmlhttpRequest(_requestData); }); } // **入口函数** function main() { autoLoadPage(); GM_registerMenuCommand("添加全局关键词", createConfigWindow); } main(); })();