您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
在 V2EX 帖子页面生成一个纯净阅读界面,支持分页合并、删除回复、复制等功能。
// ==UserScript== // @name V2EX 纯净阅读助手 // @namespace https://github.com/bitlumen // @version 2025-07-23 // @description 在 V2EX 帖子页面生成一个纯净阅读界面,支持分页合并、删除回复、复制等功能。 // @author enthem // @match https://*.v2ex.com/t/* // @icon  // @grant none // ==/UserScript== (function() { 'use strict'; const createButton = () => { const btn = document.createElement('button'); btn.textContent = '纯净阅读'; btn.style.position = 'fixed'; btn.style.bottom = '20px'; btn.style.right = '20px'; btn.style.zIndex = '99999'; btn.style.background = '#007bff'; btn.style.color = '#fff'; btn.style.padding = '8px 12px'; btn.style.border = 'none'; btn.style.borderRadius = '6px'; btn.style.cursor = 'pointer'; btn.style.boxShadow = '0 2px 5px rgba(0,0,0,0.2)'; btn.onclick = initPureReader; document.body.appendChild(btn); }; const initPureReader = async () => { const topicId = location.href.match(/t\/(\d+)/)?.[1]; if (!topicId) return alert('无法识别主题 ID'); const style = document.createElement('style'); style.innerHTML = ` #pureModal { position: fixed; inset: 0; background: rgba(0,0,0,0.6); z-index: 9999; display: flex; justify-content: center; align-items: center; } #pureModalContent { width: 80%; max-height: 90%; background: #fff; border-radius: 8px; padding: 20px; overflow-y: auto; font-family: system-ui, sans-serif; box-shadow: 0 0 10px rgba(0,0,0,0.3); position: relative; } .copyBtn { position: sticky; top: 0; right: 0; background: #007bff; color: white; padding: 6px 12px; border: none; border-radius: 4px; cursor: pointer; margin-bottom: 10px; } .copyBtn:hover { background: #0056b3; } .v2item { margin-bottom: 10px; padding-bottom: 5px; position: relative; } .v2item .meta { color: #888; font-size: 12px; margin-bottom: 4px; } .v2item .username { font-weight: bold; } .v2item .reply { margin-top: 5px; } .deleteBtn { position: absolute; top: 5px; right: 5px; background: #ccc; color: #333; border: none; padding: 2px 6px; font-size: 12px; border-radius: 4px; cursor: pointer; } .deleteBtn:hover { background: #999; } #closeBtn { position: fixed; top: 20px; left: 20px; background: #ccc; border: none; padding: 6px 10px; border-radius: 4px; cursor: pointer; } #copyToast { position: fixed; bottom: 30px; right: 30px; background: #28a745; color: white; padding: 8px 14px; border-radius: 6px; font-size: 14px; box-shadow: 0 0 5px rgba(0,0,0,0.2); z-index: 100000; animation: fadeOut 2s ease-out forwards; } @keyframes fadeOut { 0% { opacity: 1; } 80% { opacity: 1; } 100% { opacity: 0; transform: translateY(10px); } } hr { border: none; border-top: 1px solid #eee; } `; document.head.appendChild(style); const fetchPage = async (p) => { const url = `/t/${topicId}?p=${p}`; const res = await fetch(url); return await res.text(); }; const parseTime = (raw) => { const ts = new Date(raw); if (isNaN(ts)) return raw; const pad = n => n.toString().padStart(2, '0'); return `${ts.getFullYear()}-${pad(ts.getMonth()+1)}-${pad(ts.getDate())} ${pad(ts.getHours())}:${pad(ts.getMinutes())}:${pad(ts.getSeconds())}`; }; const parseContent = (html) => { const dom = new DOMParser().parseFromString(html, 'text/html'); const title = dom.querySelector('h1')?.innerText || '无标题'; const topicTimeRaw = dom.querySelector('.header .gray span[title]')?.getAttribute('title') || ''; const topicUser = dom.querySelector('.header small > a[href^="/member/"]'); const topicUserName = topicUser?.innerText || '未知'; const topicUserLink = topicUser?.href || '#'; const topicContent = dom.querySelector('.topic_content .markdown_body')?.innerHTML || ''; const replies = [...dom.querySelectorAll('.cell')].filter(cell => cell.querySelector('.reply_content')); const replyData = replies.map((cell, index) => { const user = cell.querySelector('strong > a'); const username = user?.innerText || '未知'; const userHref = user?.href || '#'; const timeRaw = cell.querySelector('.ago')?.getAttribute('title') || ''; const content = cell.querySelector('.reply_content')?.innerHTML || ''; const floor = cell.querySelector('.no')?.innerText || (index + 1); return { index: floor, username, userHref, time: parseTime(timeRaw), content }; }); return { title, topicTime: parseTime(topicTimeRaw), topicUser: { name: topicUserName, href: topicUserLink }, topicContent, replies: replyData }; }; const collectAllPages = async () => { let page = 1; const result = []; while (true) { const html = await fetchPage(page); const parsed = parseContent(html); result.push(parsed); if (!html.includes(`?p=${page + 1}`)) break; page++; } return result; }; const renderHTML = (pages) => { const wrapper = document.createElement('div'); wrapper.id = 'pureModal'; const modal = document.createElement('div'); modal.id = 'pureModalContent'; const copyBtn = document.createElement('button'); copyBtn.innerText = '📋 复制'; copyBtn.className = 'copyBtn'; const closeBtn = document.createElement('button'); closeBtn.innerText = '❌ 关闭'; closeBtn.id = 'closeBtn'; closeBtn.onclick = () => wrapper.remove(); const innerHTML = document.createElement('div'); innerHTML.id = 'pureInner'; const fragments = []; const first = pages[0]; fragments.push(` <div class="v2item" id="topic_main"> <h1 style="color: black;">${first.title}</h1> <div class="meta">作者:<a href="${first.topicUser.href}" class="username" target="_blank">${first.topicUser.name}</a> 发表于 ${first.topicTime}</div> <div class="reply">${first.topicContent}</div> </div> <hr /> `); pages.forEach((p, pageIndex) => { const replies = pageIndex === 0 ? p.replies : p.replies; replies.forEach(reply => { const id = `reply_${reply.index}_${Math.random().toString(36).slice(2)}`; const hrId = `hr_${id}`; fragments.push(` <div class="v2item" id="${id}"> <button class="deleteBtn" onclick="(function(){ document.getElementById('${id}').remove(); const hr = document.getElementById('${hrId}'); if (hr) hr.remove(); })()">❌ 删除</button> <div class="meta">${reply.index}楼 · <a href="${reply.userHref}" class="username" target="_blank">${reply.username}</a> 发表于 ${reply.time}</div> <div class="reply">${reply.content}</div> </div> <hr id="${hrId}" /> `); }); }); innerHTML.innerHTML = fragments.join('\n'); modal.appendChild(copyBtn); modal.appendChild(innerHTML); wrapper.appendChild(modal); wrapper.appendChild(closeBtn); document.body.appendChild(wrapper); copyBtn.onclick = () => { const temp = innerHTML.cloneNode(true); temp.querySelectorAll('.deleteBtn').forEach(btn => btn.remove()); const html = temp.innerHTML; const blob = new Blob([html], { type: "text/html" }); const data = [new ClipboardItem({ "text/html": blob })]; navigator.clipboard.write(data).then(() => { const toast = document.createElement('div'); toast.id = 'copyToast'; toast.innerText = '✅ 复制完成'; document.body.appendChild(toast); setTimeout(() => toast.remove(), 2000); }); }; }; const all = await collectAllPages(); renderHTML(all); }; createButton(); })();