您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
一个用于B站的用户脚本,提供快速屏蔽和管理UP主内容的功能,以及便捷的UP主访问方式
// ==UserScript== // @name bilibili-up-filter // @namespace https://github.com/sansan0/useful-userscripts // @version 3.0 // @description 一个用于B站的用户脚本,提供快速屏蔽和管理UP主内容的功能,以及便捷的UP主访问方式 // @author sansan // @match https://www.bilibili.com/* // @grant GM_addStyle // @license GPL-3.0 License // @icon  // ==/UserScript== (function () { "use strict"; // 如果当前窗口不是顶层窗口,则直接退出 if (window !== window.top) { return; } /** * 常量定义 * 定义了程序中使用的各种固定值 */ const CONSTANTS = { BLOCKED_UPS_KEY: "blocked_up_users", TOAST_DISPLAY_DURATION: 2000, TOAST_ANIMATION_DURATION: 300, HOVER_DELAY: 100, }; /** * 添加样式到页面 * 包含了所有UI组件的样式:按钮、复选框、提示框、动画等 */ function addStyles() { GM_addStyle(` /* 基础按钮样式 */ .id-block-toggle, .blocked-ups-toggle, .following-toggle { position: fixed; right: 10px; padding: 10px; border-radius: 8px; cursor: pointer; box-shadow: 0 2px 8px rgba(0,0,0,0.1); z-index: 9999; display: flex; align-items: center; justify-content: space-between; width: 100px; box-sizing: border-box; transition: all 0.3s ease; color: white; } /* ID屏蔽按钮特定样式 */ .id-block-toggle { top: calc(50% - 80px); background: #9254de; } .id-block-toggle:hover { background: #a872e8; box-shadow: 0 4px 12px rgba(0,0,0,0.2); } /* 已屏蔽按钮特定样式 */ .blocked-ups-toggle { top: 50%; transform: translateY(-50%); background: #00aeec; } .blocked-ups-toggle:hover { background: #0095cc; box-shadow: 0 4px 12px rgba(0,0,0,0.2); } /* 关注者按钮特定样式 */ .following-toggle { top: calc(50% + 40px); background: #FB7299; } .following-toggle:hover { background: #fc8bab; box-shadow: 0 4px 12px rgba(0,0,0,0.2); } /* ID屏蔽输入框样式 */ .id-block-input { position: absolute; left: 0; top: 0; width: 100%; height: 100%; background: rgba(146, 84, 222, 0.9); border: none; color: white; padding: 10px; border-radius: 8px; outline: none; display: none; font-size: 14px; box-sizing: border-box; } .id-block-input::placeholder { color: rgba(255, 255, 255, 0.8); } .id-block-toggle.input-mode .id-block-input { display: block; } .id-block-toggle.input-mode span { opacity: 0; } /* 隐藏状态样式 */ .id-block-toggle.hide, .blocked-ups-toggle.hide, .following-toggle.hide { opacity: 0; pointer-events: none; transition: opacity 0.3s ease; } /* 复选框样式 */ .block-checkbox { position: absolute; top: -40px; bottom: auto; left: 0; background: white; padding: 6px 10px; border: 1px solid #e3e5e7; border-radius: 6px; display: none; z-index: 10; box-shadow: 0 2px 8px rgba(0,0,0,0.1); transition: all 0.3s ease; font-size: 13px; color: #18191c; width: 130px; white-space: nowrap; box-sizing: border-box; pointer-events: auto; } .bili-video-card__info--owner { position: relative; z-index: 9; } .block-checkbox:hover { box-shadow: 0 4px 12px rgba(0,0,0,0.15); } .block-checkbox label { display: flex; align-items: center; cursor: pointer; user-select: none; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; width: 100%; } .block-checkbox input[type="checkbox"] { flex-shrink: 0; appearance: none; -webkit-appearance: none; width: 16px; height: 16px; border: 2px solid #c9ccd0; border-radius: 3px; margin-right: 6px; position: relative; cursor: pointer; transition: all 0.2s ease; padding: 0; } .block-checkbox input[type="checkbox"]:checked { background-color: #00aeec; border-color: #00aeec; } .block-checkbox input[type="checkbox"]:checked::after { content: ""; position: absolute; left: 4px; top: 1px; width: 5px; height: 9px; border: solid white; border-width: 0 2px 2px 0; transform: rotate(45deg); } .block-checkbox input[type="checkbox"]:hover { border-color: #00aeec; } .block-checkbox input[type="checkbox"]:focus { outline: none; box-shadow: 0 0 0 2px rgba(0,174,236,0.2); } .block-checkbox input[type="checkbox"]:disabled { background-color: #f5f5f5; border-color: #e0e0e0; cursor: not-allowed; } .block-checkbox label span { white-space: nowrap; overflow: hidden; text-overflow: ellipsis; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif; letter-spacing: 0.3px; } /* 提示框样式 */ .toast-notification { position: fixed; top: 16px; right: 16px; background: rgba(0, 0, 0, 0.8); color: white; padding: 12px 24px; border-radius: 8px; z-index: 9999999; font-size: 14px; opacity: 0; transform: translateX(100%); transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); display: flex; align-items: center; gap: 8px; pointer-events: none; box-shadow: 0 4px 12px rgba(0,0,0,0.15); } .toast-notification.show { opacity: 1; transform: translateX(0); } .toast-notification.hide { opacity: 0; transform: translateX(100%); } .toast-icon { width: 20px; height: 20px; display: flex; align-items: center; justify-content: center; font-size: 16px; } .toast-block { color: #ff6b6b; } .toast-unblock { color: #69db7c; } /* 动画效果 */ @keyframes fadeInUp { from { opacity: 0; transform: translateY(5px); } to { opacity: 1; transform: translateY(0); } } @keyframes fadeOutDown { from { opacity: 1; transform: translateY(0); } to { opacity: 0; transform: translateY(5px); } } .block-checkbox.show { animation: fadeInUp 0.3s ease forwards; } .block-checkbox.hide { animation: fadeOutDown 0.3s ease forwards; } /* 面板样式 */ .blocked-ups-panel, .following-panel { position: fixed; right: -350px; top: 0; width: 320px; height: 100vh; background: white; box-shadow: -2px 0 8px rgba(0,0,0,0.1); z-index: 9998; transition: right 0.3s ease; display: flex; flex-direction: column; } .blocked-ups-panel.show, .following-panel.show { right: 0; } .blocked-ups-header { padding: 16px; background: #f6f7f8; border-bottom: 1px solid #e3e5e7; font-weight: bold; display: flex; justify-content: space-between; align-items: center; } .blocked-ups-list { flex: 1; overflow-y: auto; padding: 12px; } .blocked-up-item { display: flex; justify-content: space-between; align-items: center; padding: 10px; border-bottom: 1px solid #f0f1f2; transition: background-color 0.2s ease; } .blocked-up-item:hover { background-color: #f6f7f8; } .blocked-up-info { flex: 1; min-width: 0; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; color: #18191c; text-decoration: none; margin-right: 10px; } .blocked-up-remove { padding: 4px 8px; border: none; background: #ff6b6b; color: white; border-radius: 4px; cursor: pointer; margin-left: 8px; transition: background-color 0.2s ease; } .blocked-up-remove:hover { background: #ff5252; } .follow-time { color: #99a2aa; font-size: 12px; margin-left: auto; padding-left: 10px; white-space: nowrap; } /* 滚动条样式 */ .blocked-ups-list::-webkit-scrollbar { width: 6px; } .blocked-ups-list::-webkit-scrollbar-track { background: #f1f1f1; } .blocked-ups-list::-webkit-scrollbar-thumb { background: #c0c0c0; border-radius: 3px; } .blocked-ups-list::-webkit-scrollbar-thumb:hover { background: #a0a0a0; } /* 移除动画 */ @keyframes removeItem { 0% { opacity: 1; transform: translateX(0); } 100% { opacity: 0; transform: translateX(100%); height: 0; padding: 0; margin: 0; } } .blocked-up-item.removing { animation: removeItem 0.3s ease-out forwards; overflow: hidden; } /* Toast动画 */ @keyframes toastIn { from { transform: translateX(100%); opacity: 0; } to { transform: translateX(0); opacity: 1; } } @keyframes toastOut { from { transform: translateX(0); opacity: 1; } to { transform: translateX(100%); opacity: 0; } } .toast-notification.show { animation: toastIn 0.3s cubic-bezier(0.4, 0, 0.2, 1) forwards; } .toast-notification.hide { animation: toastOut 0.3s cubic-bezier(0.4, 0, 0.2, 1) forwards; } `); } /** * 数据存储与管理模块 * 处理与localStorage相关的所有操作 */ const DataManager = { /** * 从localStorage获取被屏蔽的UP主列表 * @returns {Array} 返回被屏蔽UP主的数组,每个元素包含id和name */ getBlockedUps() { try { const blockedUps = localStorage.getItem(CONSTANTS.BLOCKED_UPS_KEY); return blockedUps ? JSON.parse(blockedUps) : []; } catch (error) { console.error("Error getting blocked UPs:", error); return []; } }, /** * 添加UP主到屏蔽列表 * @param {string} upId - UP主的ID * @param {string} upName - UP主的昵称 */ addBlockedUp(upId, upName) { const blockedUps = this.getBlockedUps(); const existingIndex = blockedUps.findIndex((up) => up.id === upId); if (existingIndex > -1 && !blockedUps[existingIndex].name) { blockedUps[existingIndex].name = upName; } else if (existingIndex === -1) { blockedUps.push({ id: upId, name: upName }); } localStorage.setItem( CONSTANTS.BLOCKED_UPS_KEY, JSON.stringify(blockedUps) ); if (document.body.hasAttribute("data-panel-initialized")) { UIManager.updateBlockedUpsList(); } }, /** * 从屏蔽列表中移除指定UP主 * @param {string} upId - 需要移除的UP主ID */ removeBlockedUp(upId) { const blockedUps = this.getBlockedUps(); const filteredUps = blockedUps.filter((up) => up.id !== upId); localStorage.setItem( CONSTANTS.BLOCKED_UPS_KEY, JSON.stringify(filteredUps) ); if (document.body.hasAttribute("data-panel-initialized")) { UIManager.updateBlockedUpsList(); } }, /** * 检查UP主是否被屏蔽 * @param {string} upId - UP主ID * @returns {boolean} 如果被屏蔽返回true,否则返回false */ isBlocked(upId) { return this.getBlockedUps().some((up) => up.id === upId); }, }; /** * UI管理模块 * 处理所有界面元素的创建、更新和交互 */ const UIManager = { blockedUpsPanel: null, /** * 显示操作提示框 * @param {string} upName - UP主昵称 * @param {boolean} isBlocking - true表示屏蔽操作,false表示取消屏蔽 */ showToast(upName, isBlocking) { const existingToast = document.querySelector(".toast-notification"); if (existingToast) existingToast.remove(); const toast = document.createElement("div"); toast.className = "toast-notification"; const icon = isBlocking ? "🚫" : "✅"; const action = isBlocking ? "已屏蔽" : "已取消屏蔽"; toast.innerHTML = ` <span class="toast-icon ${ isBlocking ? "toast-block" : "toast-unblock" }">${icon}</span> <span>${action} UP主:${upName}</span> `; document.body.appendChild(toast); toast.offsetHeight; // 触发重绘 requestAnimationFrame(() => toast.classList.add("show")); setTimeout(() => { toast.classList.add("hide"); setTimeout(() => toast.remove(), CONSTANTS.TOAST_ANIMATION_DURATION); }, CONSTANTS.TOAST_DISPLAY_DURATION); }, /** * 更新被屏蔽UP主列表显示 */ updateBlockedUpsList() { const panel = document.querySelector(".blocked-ups-panel"); const listContainer = panel?.querySelector(".blocked-ups-list"); if (!listContainer) return; const blockedUps = DataManager.getBlockedUps(); const reversedUps = [...blockedUps].reverse(); listContainer.innerHTML = reversedUps .map( (up) => ` <div class="blocked-up-item" data-up-id="${up.id}"> <a href="https://space.bilibili.com/${up.id}" target="_blank" class="blocked-up-info"> ${up.name || `ID: ${up.id}`} </a> <button class="blocked-up-remove">移除</button> </div> ` ) .join(""); const toggleButton = document.querySelector(".blocked-ups-toggle"); if (toggleButton) { toggleButton.querySelector("span:last-child").textContent = blockedUps.length; } const countElement = panel.querySelector(".blocked-count"); if (countElement) { countElement.textContent = blockedUps.length; } }, /** * 创建UP主屏蔽控制的复选框 * @param {string} upId - UP主ID * @param {string} upName - UP主昵称 * @returns {HTMLElement} 返回创建的复选框元素 */ createCheckbox(upId, upName) { const checkbox = document.createElement("div"); checkbox.className = "block-checkbox"; const isBlocked = DataManager.isBlocked(upId); checkbox.innerHTML = ` <label> <input type="checkbox" ${isBlocked ? "checked" : ""}> <span>屏蔽该UP主</span> </label> `; const input = checkbox.querySelector("input"); input.addEventListener("change", (e) => { if (e.target.checked) { DataManager.addBlockedUp(upId, upName); this.showToast(upName, true); } else { DataManager.removeBlockedUp(upId); this.updateCheckboxStatus(upId); this.showToast(upName, false); } ContentFilter.filterBlockedContent(); }); return checkbox; }, /** * 更新所有相关复选框的状态 * @param {string} upId - UP主ID */ updateCheckboxStatus(upId) { document .querySelectorAll(".bili-video-card__info--owner") .forEach((link) => { const href = link.getAttribute("href"); if (!href) return; const currentUpId = href.match(/\/(\d+)$/)?.[1]; if (currentUpId === upId) { const checkbox = link.querySelector( '.block-checkbox input[type="checkbox"]' ); if (checkbox) { checkbox.checked = false; } } }); }, /** * 处理UP主链接的鼠标悬停事件 * @param {HTMLElement} element - UP主链接元素 */ handleHover(element) { const href = element.getAttribute("href"); if (!href) return; const upId = href.match(/\/(\d+)$/)?.[1]; if (!upId) return; const authorSpan = element.querySelector( ".bili-video-card__info--author" ); const upName = authorSpan?.getAttribute("title") || "未知UP主"; let checkbox = element.querySelector(".block-checkbox"); if (!checkbox) { checkbox = this.createCheckbox(upId, upName); element.style.position = "relative"; element.appendChild(checkbox); } else { const input = checkbox.querySelector('input[type="checkbox"]'); const isBlocked = DataManager.isBlocked(upId); input.checked = isBlocked; } let hideTimeout; let isOverElement = false; let isOverCheckbox = false; const showCheckbox = () => { checkbox.style.display = "block"; requestAnimationFrame(() => { checkbox.style.opacity = "1"; checkbox.style.transform = "translateY(0)"; }); }; const hideCheckbox = () => { checkbox.style.opacity = "0"; checkbox.style.transform = "translateY(5px)"; hideTimeout = setTimeout(() => { checkbox.style.display = "none"; }, CONSTANTS.TOAST_ANIMATION_DURATION); }; const checkVisibility = () => { setTimeout(() => { if (!isOverElement && !isOverCheckbox) { hideCheckbox(); } }, CONSTANTS.HOVER_DELAY); }; element.addEventListener("mouseenter", () => { isOverElement = true; clearTimeout(hideTimeout); showCheckbox(); }); element.addEventListener("mouseleave", () => { isOverElement = false; checkVisibility(); }); checkbox.addEventListener("mouseenter", () => { isOverCheckbox = true; clearTimeout(hideTimeout); }); checkbox.addEventListener("mouseleave", () => { isOverCheckbox = false; checkVisibility(); }); element.setAttribute("data-hover-initialized", "true"); }, /** * 切换所有按钮的显示/隐藏状态 * @param {boolean} hide - 为true时隐藏按钮,为false时显示按钮 */ toggleAllButtons(hide = true) { const followingToggle = document.querySelector(".following-toggle"); const blockedToggle = document.querySelector(".blocked-ups-toggle"); const idBlockToggle = document.querySelector(".id-block-toggle"); if (hide) { followingToggle?.classList.add("hide"); blockedToggle?.classList.add("hide"); idBlockToggle?.classList.add("hide"); } else { followingToggle?.classList.remove("hide"); blockedToggle?.classList.remove("hide"); idBlockToggle?.classList.remove("hide"); } }, /** * 创建并管理被屏蔽UP主列表面板 * @returns {Object} 返回包含更新面板方法的对象 */ createBlockedUpsPanel() { // 创建ID屏蔽按钮 const idBlockButton = document.createElement("div"); idBlockButton.className = "id-block-toggle"; idBlockButton.innerHTML = ` <span>手动屏蔽ID</span> <span></span> <input type="text" class="id-block-input" placeholder="输入UP主ID" pattern="[0-9]*"> `; // 获取输入框元素 const idInput = idBlockButton.querySelector(".id-block-input"); // 处理ID输入框事件 this._setupIdInputEvents(idBlockButton, idInput); document.body.appendChild(idBlockButton); // 创建已屏蔽按钮 const toggleButton = document.createElement("div"); toggleButton.className = "blocked-ups-toggle"; toggleButton.innerHTML = ` <span>已屏蔽</span> <span>${DataManager.getBlockedUps().length}</span> `; // 创建屏蔽列表面板 const panel = document.createElement("div"); panel.className = "blocked-ups-panel"; panel.innerHTML = ` <div class="blocked-ups-header"> <span>已屏蔽的UP主</span> <span class="blocked-count">${ DataManager.getBlockedUps().length }</span> </div> <div class="blocked-ups-list"></div> `; // 设置面板交互 this._setupPanelInteraction(toggleButton, panel); this._setupRemoveButtons(panel); document.body.appendChild(toggleButton); document.body.appendChild(panel); this.updateBlockedUpsList(); document.body.setAttribute("data-panel-initialized", "true"); return { updateBlockedUpsList: () => this.updateBlockedUpsList(), }; }, /** * 设置ID输入框的事件处理 * @private * @param {HTMLElement} idBlockButton - ID屏蔽按钮元素 * @param {HTMLElement} idInput - 输入框元素 */ _setupIdInputEvents(idBlockButton, idInput) { // 处理鼠标进入事件 idBlockButton.addEventListener("mouseenter", () => { idBlockButton.classList.add("input-mode"); idInput.focus(); }); // 处理鼠标离开事件 idBlockButton.addEventListener("mouseleave", () => { if (document.activeElement !== idInput) { idBlockButton.classList.remove("input-mode"); idInput.value = ""; } }); // 处理输入事件 idInput.addEventListener("input", (e) => { e.target.value = e.target.value.replace(/\D/g, ""); }); // 处理键盘事件 idInput.addEventListener("keydown", (e) => { if (e.key === "Enter") { const upId = e.target.value.trim(); if (upId) { DataManager.addBlockedUp(upId, null); this.showToast(`ID: ${upId}`, true); e.target.value = ""; idBlockButton.classList.remove("input-mode"); this.updateBlockedUpsList(); ContentFilter.filterBlockedContent(); } } else if (e.key === "Escape") { idBlockButton.classList.remove("input-mode"); e.target.value = ""; idBlockButton.blur(); } }); // 处理失去焦点事件 idInput.addEventListener("blur", () => { setTimeout(() => { if (document.activeElement !== idInput) { idBlockButton.classList.remove("input-mode"); idInput.value = ""; } }, 200); }); }, /** * 设置面板交互事件 * @private * @param {HTMLElement} toggleButton - 触发按钮 * @param {HTMLElement} panel - 面板元素 */ _setupPanelInteraction(toggleButton, panel) { let mouseInPanel = false; toggleButton.addEventListener("mouseenter", () => { panel.classList.add("show"); this.toggleAllButtons(true); this.updateBlockedUpsList(); }); panel.addEventListener("mouseenter", () => { mouseInPanel = true; }); panel.addEventListener("mouseleave", () => { mouseInPanel = false; setTimeout(() => { if (!mouseInPanel && !toggleButton.matches(":hover")) { panel.classList.remove("show"); this.toggleAllButtons(false); } }, 200); }); toggleButton.addEventListener("mouseleave", () => { setTimeout(() => { if (!mouseInPanel) { panel.classList.remove("show"); this.toggleAllButtons(false); } }, 200); }); }, /** * 设置移除按钮的事件处理 * @private * @param {HTMLElement} panel - 面板元素 */ _setupRemoveButtons(panel) { panel.addEventListener("click", (e) => { if (e.target.classList.contains("blocked-up-remove")) { const upItem = e.target.closest(".blocked-up-item"); if (!upItem) return; const upId = upItem.dataset.upId; const upName = upItem .querySelector(".blocked-up-info") .textContent.trim(); upItem.classList.add("removing"); setTimeout(() => { DataManager.removeBlockedUp(upId); this.updateCheckboxStatus(upId); this.showToast(upName, false); this.updateBlockedUpsList(); ContentFilter.filterBlockedContent(); }, 300); } }); }, /** * 添加屏蔽复选框到UP主头像 */ addCheckboxToUpAvatar() { const upAvatar = document.querySelector("a.up-avatar"); if (!upAvatar) return; const href = upAvatar.getAttribute("href"); if (!href) return; const upId = href.match(/\/(\d+)$/)?.[1]; if (!upId) return; const upNameElement = document.querySelector(".up-name"); const upName = upNameElement?.textContent?.trim() || "未知UP主"; const createAndAddCheckbox = () => { let checkbox = upAvatar.querySelector(".block-checkbox"); if (!checkbox) { checkbox = this.createCheckbox(upId, upName); upAvatar.style.position = "relative"; upAvatar.appendChild(checkbox); checkbox.style.display = "block"; checkbox.style.opacity = "1"; checkbox.style.transform = "translateY(0)"; } }; // 立即创建一次 createAndAddCheckbox(); // 创建专门监听 up-avatar 的观察器 const avatarObserver = new MutationObserver(() => { createAndAddCheckbox(); }); // 配置观察器 avatarObserver.observe(upAvatar, { childList: true, // 监听子元素变化 subtree: true, // 监听所有后代元素 characterData: true, // 监听文本内容变化 attributes: true, // 监听属性变化 }); }, /** * 创建并管理关注列表面板 * @returns {Promise<void>} */ async createFollowingPanel() { if (this.followingPanelInitialized) return; try { const statsContainer = await DOMUtils.waitForElement( ".logged-in > .stats" ); if (!statsContainer) return; const followLink = statsContainer.querySelector("a"); if (!followLink) { console.warn("未找到关注链接"); return; } const userId = followLink.href.match( /space\.bilibili\.com\/(\d+)/ )?.[1]; if (!userId) { console.warn("无法获取用户ID"); return; } const followCount = followLink.querySelector(".stats-number")?.textContent; const followTotal = parseInt(followCount) || 0; const toggleButton = document.createElement("div"); toggleButton.className = "following-toggle"; toggleButton.innerHTML = ` <span>关注者</span> <span>${followTotal}</span> `; const panel = document.createElement("div"); panel.className = "following-panel"; panel.innerHTML = ` <div class="blocked-ups-header"> <span>关注的UP主</span> <span class="following-count">${ followTotal > 50 ? "最近 50 位" : followTotal }</span> </div> <div class="blocked-ups-list"></div> `; this._setupFollowingPanelEvents(toggleButton, panel, userId); document.body.appendChild(toggleButton); document.body.appendChild(panel); this.followingPanelInitialized = true; } catch (error) { console.warn("创建关注者面板失败:", error); } }, /** * 设置关注面板的事件处理和数据加载 * @private * @param {HTMLElement} toggleButton - 触发按钮 * @param {HTMLElement} panel - 面板元素 * @param {string} userId - 用户ID */ _setupFollowingPanelEvents(toggleButton, panel, userId) { let mouseInPanel = false; // 获取关注列表 const updateFollowingList = async () => { const listContainer = panel.querySelector(".blocked-ups-list"); const data = await APIManager.fetchFollowingList(userId); if (!data) return; const headerText = panel.querySelector( ".blocked-ups-header span:first-child" ); headerText.textContent = "关注的UP主"; listContainer.innerHTML = data.list .map( (up) => ` <div class="blocked-up-item"> <a href="https://space.bilibili.com/${up.mid}" target="_blank" class="blocked-up-info"> ${up.uname} </a> <span class="follow-time">${DOMUtils.formatDate( up.mtime )}</span> </div> ` ) .join(""); }; // 设置面板交互 toggleButton.addEventListener("mouseenter", () => { panel.classList.add("show"); this.toggleAllButtons(true); updateFollowingList(); }); panel.addEventListener("mouseenter", () => { mouseInPanel = true; }); panel.addEventListener("mouseleave", () => { mouseInPanel = false; setTimeout(() => { if (!mouseInPanel && !toggleButton.matches(":hover")) { panel.classList.remove("show"); this.toggleAllButtons(false); } }, 200); }); toggleButton.addEventListener("mouseleave", () => { setTimeout(() => { if (!mouseInPanel) { panel.classList.remove("show"); this.toggleAllButtons(false); } }, 200); }); }, }; /** * 内容过滤器模块 * 负责屏蔽和过滤网页上的内容 */ const ContentFilter = { /** * 过滤并移除被屏蔽UP主的内容 */ filterBlockedContent() { // 移除特定类型的卡片 document .querySelectorAll(".floor-single-card, .bili-live-card") .forEach((card) => card.remove()); const blockedUps = DataManager.getBlockedUps(); const filteredUps = new Set(); document .querySelectorAll(".bili-video-card__info--owner") .forEach((link) => { const upId = link.getAttribute("href")?.match(/\/(\d+)$/)?.[1]; if (upId && blockedUps.some((up) => up.id === upId)) { const authorSpan = link.querySelector( ".bili-video-card__info--author" ); const upName = authorSpan?.getAttribute("title") || "未知UP主"; DataManager.addBlockedUp(upId, upName); const targetCard = link.closest(".feed-card") || link.closest(".bili-video-card"); if (targetCard && !filteredUps.has(upId)) { targetCard.remove(); UIManager.showToast(upName, true); filteredUps.add(upId); } } }); }, /** * 初始化MutationObserver以监视DOM变化并过滤内容 */ initContentObserver() { const observer = new MutationObserver((mutations) => { mutations.forEach((mutation) => { mutation.addedNodes.forEach((node) => { if (node.nodeType === 1) { // 移除不需要的卡片 if ( node.classList?.contains("floor-single-card") || node.classList?.contains("bili-live-card") ) { node.remove(); return; } // 移除子节点中的卡片 node .querySelectorAll?.(".floor-single-card, .bili-live-card") .forEach((card) => card.remove()); // 处理视频卡片中的UP主链接 node .querySelectorAll?.(".bili-video-card__info--owner") .forEach((link) => { if (!link.getAttribute("data-hover-initialized")) { UIManager.handleHover(link); } }); } }); this.filterBlockedContent(); }); }); // 启动观察器 observer.observe(document.body, { childList: true, subtree: true, }); }, }; /** * API管理模块 * 处理与外部API的交互 */ const APIManager = { /** * 获取用户关注列表 * @param {string} userId - 用户ID * @returns {Promise<Object|null>} 关注列表数据或null(出错时) */ async fetchFollowingList(userId) { try { // 无论总数多少,始终只请求最近的50个 const response = await fetch( `https://api.bilibili.com/x/relation/followings?vmid=${userId}&ps=50`, { credentials: "include", headers: { Accept: "application/json" }, } ); if (!response.ok) throw new Error(`HTTP error! status: ${response.status}`); const data = await response.json(); if (data.code !== 0) throw new Error( `API error! code: ${data.code}, message: ${data.message}` ); return data.data; } catch (error) { console.error("Error fetching following list:", error); return null; } }, }; /** * DOM工具模块 * 提供DOM操作相关的辅助函数 */ const DOMUtils = { /** * 等待指定元素加载完成 * @param {string} selector - 元素选择器 * @param {number} maxAttempts - 最大尝试次数 * @param {number} interval - 尝试间隔(毫秒) * @param {number} initialDelay - 初始延迟(毫秒) * @returns {Promise<HTMLElement|null>} 返回找到的元素或null */ waitForElement( selector, maxAttempts = 3, interval = 1000, initialDelay = 1000 ) { return new Promise((resolve, reject) => { let attempts = 0; const checkElement = () => { const element = document.querySelector(selector); if (element) { resolve(element); return; } attempts++; if (attempts >= maxAttempts) { reject( new Error( `Element ${selector} not found after ${maxAttempts} attempts` ) ); return; } setTimeout(checkElement, interval); }; setTimeout(checkElement, initialDelay); }); }, /** * 格式化时间戳为日期字符串 * @param {number} timestamp - 时间戳(秒) * @returns {string} 格式化后的日期字符串(YYYY-MM-DD) */ formatDate(timestamp) { const date = new Date(timestamp * 1000); // 转换为毫秒 const year = date.getFullYear(); const month = String(date.getMonth() + 1).padStart(2, "0"); const day = String(date.getDate()).padStart(2, "0"); return `${year}-${month}-${day}`; }, }; /** * 初始化函数 * 程序入口点,初始化各个组件 */ function initialize() { // 添加样式 addStyles(); // 初始化视频卡片上的UP主触发器 document .querySelectorAll(".bili-video-card__info--owner") .forEach((element) => UIManager.handleHover(element)); // 添加UP主头像复选框 UIManager.addCheckboxToUpAvatar(); // 过滤屏蔽内容 ContentFilter.filterBlockedContent(); // 初始化内容观察器 ContentFilter.initContentObserver(); // 页面加载完成后创建面板 window.addEventListener("load", async () => { await UIManager.createFollowingPanel(); UIManager.blockedUpsPanel = UIManager.createBlockedUpsPanel(); }); } // 启动程序 initialize(); })();