继续操作前请注册或者登录。
对 linux.do 的增强脚本
// ==UserScript== // @name Linux.po // @namespace http://tampermonkey.net/ // @version 0.1.8 // @description 对 linux.do 的增强脚本 // @author PRO-2684 // @match https://linux.do/* // @run-at document-start // @icon ###Rak9cY/x/G9pa4SNlgBs7AAAAABJRU5ErkJggg== // @license gpl-3.0 // @grant GM_setValue // @grant GM_getValue // @grant GM_deleteValue // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_addValueChangeListener // @require https://github.com/PRO-2684/GM_config/releases/download/v1.2.1/config.min.js#md5=525526b8f0b6b8606cedf08c651163c2 // ==/UserScript== (function () { 'use strict'; const { name, version } = GM_info.script; const idPrefix = "linux-po-"; const configDesc = { $default: { autoClose: false, }, appearance: { name: "🎨 外观", title: "外观", type: "folder", items: { sidebarManager: { name: "⬅️ 侧栏管理", title: "允许你隐藏侧栏中的各个部分", type: "folder", items: { $default: { value: false, input: "current", processor: "not", title: (prop, value, desc) => desc.name, formatter: (prop, value, desc) => `${desc.name}: ${value ? "🫥" : "👀"}`, }, customCategories: { name: "自定义板块" }, externalLinks: { name: "外部链接" }, categories: { name: "类别" }, tags: { name: "标签" }, messages: { name: "消息" }, channels: { name: "频道" }, directMessages: { name: "直接消息" }, chat: { name: "聊天" }, bottomMenu: { name: "底部菜单" }, }, }, postManager: { name: "📝 帖子管理", title: "允许你隐藏帖子的各个部分", type: "folder", items: { $default: { value: false, input: "current", processor: "not", title: (prop, value, desc) => desc.name, formatter: (prop, value, desc) => `${desc.name}: ${value ? "🫥" : "👀"}`, }, secondaryName: { name: "次要名称" }, userTitle: { name: "头衔" }, userStatus: { name: "自定义状态" }, posterIcon: { name: "🍰", title: "加入社区纪念日以及生日图标 (discourse-cakeday)", }, postNotice: { name: "帖子通知", title: "新用户首次发言、回归用户等标识", }, flair: { name: "资质", title: "展示在头像右下角", }, }, }, }, }, accessibility: { name: "♿ 辅助功能", title: "辅助功能", type: "folder", items: { largerClickArea: { name: "👆 增大点击区域", title: "增大帖子列表中各帖子的可点击区域 (仅支持左键)", type: "bool", value: true, }, showPostsFloor: { name: "🔢 显示楼层", title: "在帖子中显示楼层", type: "bool", value: false, }, atBeforeUsername: { name: "👤 @用户名", title: "在用户名前添加 @ 符号", type: "bool", value: false, }, }, }, }; const config = new GM_config(configDesc); // Helper function for css function injectCSS(id, css) { const style = document.head.appendChild(document.createElement("style")); style.id = idPrefix + id; style.textContent = css; return style; } function cssHelper(id, enable) { const current = document.getElementById(idPrefix + id); if (current) { current.disabled = !enable; } else if (enable) { injectCSS(id, dynamicStyles[id]); } } /** * Generates CSS for hiding given sidebar section. */ function hideSidebarSection(section) { return `#d-sidebar > .sidebar-sections div.sidebar-section[data-section-name="${section}"] { display: none; }`; } /** * Generates CSS for hiding given post section. */ function hidePostSection(section) { return `.post-stream > .topic-post > article .names > .${section} { display: none; }`; } // Dynamic styles const dynamicStyles = { "appearance.sidebarManager.customCategories": hideSidebarSection("community"), "appearance.sidebarManager.externalLinks": hideSidebarSection("外部链接"), "appearance.sidebarManager.categories": hideSidebarSection("categories"), "appearance.sidebarManager.tags": hideSidebarSection("tags"), "appearance.sidebarManager.messages": hideSidebarSection("messages"), "appearance.sidebarManager.channels": hideSidebarSection("chat-channels"), "appearance.sidebarManager.directMessages": hideSidebarSection("chat-dms"), "appearance.sidebarManager.chat": "#d-sidebar > button[data-key='chat'] { display: none; }", "appearance.sidebarManager.bottomMenu": "#d-sidebar > div.sidebar-footer-wrapper { display: none; }", "appearance.postManager.secondaryName": hidePostSection("second"), "appearance.postManager.userTitle": hidePostSection("user-title"), "appearance.postManager.userStatus": hidePostSection("user-status-message-wrap"), "appearance.postManager.posterIcon": hidePostSection("poster-icon"), "appearance.postManager.flair": ".topic-avatar > .post-avatar > .avatar-flair { display: none; }", "appearance.postManager.postNotice": ".post-stream > .topic-post > article .post-notice { display: none; }", "accessibility.largerClickArea": ".topic-list-item > .main-link { cursor: pointer; }", "accessibility.showPostsFloor": `.post-stream > .topic-post > article[id^='post_'] { &::after { content: attr(id) '#'; color: var(--primary-med-or-secondary-med); position: absolute; right: 0; top: calc(0.8em + 1px); text-indent: -2.5em; overflow: hidden; /* Dirty trick to hide leading "post_" */ } .embedded-posts > .reply .post-link-arrow > a.post-info::after { content: attr(href) '#'; display: inline-flex; text-indent: -7.9em; overflow: hidden; /* Dirty trick to hide leading "/t/topic/\\d{6}/" */ } } .timeline-container > .topic-timeline > .timeline-scrollarea-wrapper > .timeline-date-wrapper > .now-date[href^='/t/topic/']::after { content: attr(href) '#'; display: inline-flex; margin-left: 0.2em; text-indent: -7.9em; overflow: hidden; /* Dirty trick to hide leading "/t/topic/\\d{6}/" */ }`, "accessibility.atBeforeUsername": ` span.username > a::before { content: "@"; } div.username::before { content: "@"; } `, }; for (const prop in dynamicStyles) { cssHelper(prop, config.get(prop)); } // Accessibility // Larger click area let largerClickAreaEnabled = false; /** * Handles the click event when larger click area is enabled. * @param {MouseEvent} e */ function largerClickAreaHandler(e) { if (e.defaultPrevented || !e.isTrusted) return; const mainLink = e.target.closest(".topic-list-item > .main-link"); if (mainLink) { e.preventDefault(); const title = mainLink.querySelector(".title.raw-link.raw-topic-link"); title?.click(); } } /** * Enables or disables the larger click area feature. * @param {boolean} enable */ function largerClickArea(enable) { if (enable && !largerClickAreaEnabled) { document.body.addEventListener("click", largerClickAreaHandler); largerClickAreaEnabled = true; } else if (!enable && largerClickAreaEnabled) { document.body.removeEventListener("click", largerClickAreaHandler); largerClickAreaEnabled = false; } } // Callbacks const callbacks = { "accessibility.largerClickArea": largerClickArea, }; for (const [prop, callback] of Object.entries(callbacks)) { callback(config.get(prop)); } config.addEventListener("set", e => { if (e.detail.prop in dynamicStyles) { cssHelper(e.detail.prop, e.detail.after); } if (e.detail.prop in callbacks) { callbacks[e.detail.prop](e.detail.after); } }); // https://linux.do/emojis.json })();