V2EX 纯净阅读助手

在 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();

})();