Yamibo 漫画阅读器

一键进入漫画阅读器,支持自动生成系列目录,智能匹配章节标题。

当前为 2025-11-22 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         Yamibo 漫画阅读器
// @namespace    https://bbs.yamibo.com/
// @version      3.6.2
// @author       hitori酱
// @description  一键进入漫画阅读器,支持自动生成系列目录,智能匹配章节标题。
// @match        https://bbs.yamibo.com/thread-*
// @match        https://bbs.yamibo.com/forum.php?mod=viewthread&*
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_deleteValue
// @grant        GM_listValues
// @connect      bbs.yamibo.com
// @require      https://cdn.jsdelivr.net/npm/[email protected]
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/umd/cn2t.js
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/umd/t2cn.js
// @run-at       document-start
// @noframes
// @license      MIT License
// @icon         
// ==/UserScript==

(function () {
  "use strict";
  // 旧版本缓存清理
  const currentVersion = GM_info.script.version;
  const lastVersion = GM_getValue("script_version", "0.0.0");
  if (currentVersion !== lastVersion) {
    console.log(`检测到脚本更新: 从 v${lastVersion} 更新到 v${currentVersion}`);
    const keys = GM_listValues();
    keys.forEach((key) => {
      if (key !== "script_version") {
        // 保留版本号字段
        GM_deleteValue(key);
      }
    });
    console.log("已清除旧版本缓存数据");
    GM_setValue("script_version", currentVersion);
  }

  // 静态常量预编译
  const EXCLUDE_WORDS_REGEX =
    /个人汉化|汉化组|汉化|个人翻译|個人翻譯|个汉|提灯喵|翻译|生肉|未翻译|填坑组|粮食组|保护协会|創作百合|创作百合|熟肉|字幕|工作室|社团|官方中字|出版|同人志|汉化工房|漫画屋|同好會|翻译组|汉化委员会|渣翻渣嵌|合作汉化|完结|连载|短篇|短篇合集|百合短篇合集|同人|raw|韩漫|杂志|英译|单行本|插画|特典|番外|外传|彩页|合订本|猫岛汉化组|和菓子漫画屋|大友同好會|透明声彩汉化组|最终|终章|序章|尾声/i;
  const FATAL_KEYWORDS_REGEX =
    /汉化|翻译|个人|组|工作室|社团|同人|英译|中字|生肉|raw|sample|合同志|填坑|粮食|发布|校对|嵌字|图源|扫图|合集|短篇|短篇集|连载/i;
  const BLOCK_IMG_REGEX =
    /\/uc_server\/data\/avatar\/|avatar|user_avatar|usericon|static\/image\/common\/|static\/image\/smiley\/|template\/|none\.gif|loading\.gif|logo\.png|logo\.gif|qq\.gif|qq_big\.gif|qq_group|userinfo\.gif|forumlink\.gif|online_admin|online_member|online_team|icon_quote|collapse|expand|rating|score|grade|star|magic/i;
  const NUM_CHAR = "[\\d①-⑳一二三四五六七八九十百千万零〇]";
  const NUM_RANGE_UNIT = `(?:${NUM_CHAR}|[-~—–])`;

  // 样式常量预编译
  const BUTTON_CSS = `
    /* === 入口按钮 === */
    #reader-toggle {
      position: fixed; top: 20%; right: 10px; transform: translateY(-50%);
      z-index: 99999; width: 60px; height: 60px; border-radius: 60px;
      background: white; border: none; cursor: pointer;
      box-shadow: 0 10px 25px rgba(0,0,0,0.1);
      display: flex; justify-content: center; align-items: center;
      transition: 0.5s;
      font-family: "IBM Plex Sans SC", "PingFang SC", "Microsoft YaHei", Arial, sans-serif !important;
    }
    #reader-toggle:hover { width: 180px; box-shadow: none; }

    /* 渐变背景层 */
    #reader-toggle::before, #reader-toggle::after {
      content: ""; position: absolute; border-radius: 40px;
      background: linear-gradient(45deg, #56CCF2, #2F80ED); opacity: 0; transition: 0.5s;
    }
    #reader-toggle::before { inset: 0; }
    #reader-toggle::after { top: 10px; width: 100%; height: 100%; filter: blur(15px); z-index: -1; }

    #reader-toggle:hover::before { opacity: 1; }
    #reader-toggle:hover::after { opacity: 0.5; }

    /* 图标与文字 */
    #reader-toggle .icon {
      position: absolute; left: 0; top: 0; width: 60px; height: 60px;
      display: flex; align-items: center; justify-content: center;
      font-size: 2em; color: #777; transition: 0.5s;
    }
    #reader-toggle .icon ion-icon { width: 24px; height: 24px; }

    #reader-toggle .title {
      position: absolute; color: #fff; font-size: 1.4em; letter-spacing: 0.1em;
      transform: scale(0); transition: 0.5s;
    }

    #reader-toggle:hover .icon { transform: scale(0); }
    #reader-toggle:hover .title { transform: scale(1); transition-delay: 0.2s; }

    /* === 目录侧边栏 === */
    #directory-sidebar {
      position: fixed; top: 0; right: -400px; width: 400px; height: 100%;
      background: #1e293b; /* 纯色性能更好,且配合阅读器暗色 */
      z-index: 2147483647; transition: right 0.3s cubic-bezier(0.4, 0, 0.2, 1);
      box-shadow: -15px 0 40px rgba(0,0,0,0.4);
      display: flex; flex-direction: column; color: white; border: none;
    }
    #directory-sidebar.open { right: 0; }

    /* 侧边栏头部 */
    #directory-header {
      display: flex; justify-content: space-between; align-items: center;
      padding: 15px 20px; border-bottom: 1px solid rgba(255,255,255,0.1);
    }
    #directory-title { font-size: 16px; font-weight: 600; letter-spacing: 0.5px; }
    #directory-close {
      background: rgba(255,255,255,0.1); border: none; color: white; cursor: pointer;
      width: 32px; height: 32px; border-radius: 6px; font-size: 18px;
      display: flex; align-items: center; justify-content: center; transition: 0.2s;
    }
    #directory-close:hover { background: rgba(255,255,255,0.2); transform: scale(1.1); }

    /* 目录列表内容 */
    #directory-content {
      flex: 1; overflow-y: auto; padding: 10px 0;
      scrollbar-width: thin; scrollbar-color: rgba(255,255,255,0.3) transparent;
    }
    #directory-content::-webkit-scrollbar { width: 6px; }
    #directory-content::-webkit-scrollbar-thumb { background: rgba(255,255,255,0.3); border-radius: 3px; }

    #directory-list { list-style: none; padding: 0 16px; margin: 0; }

    .directory-item {
      display: block; padding: 12px 16px; margin-bottom: 8px;
      background: rgba(255,255,255,0.05); border-radius: 8px;
      color: rgba(255,255,255,0.8); text-decoration: none !important;
      font-size: 14px; transition: all 0.2s; cursor: pointer; position: relative; overflow: hidden;
    }

    /* 悬停高亮 */
    .directory-item:hover {
      background: rgba(255,255,255,0.1); color: #fff; transform: translateX(5px);
    }
    /* 当前章节 */
    .directory-item.current {
      background: linear-gradient(135deg, #FF6B6B, #FF8E53);
      color: white; font-weight: 600; box-shadow: 0 4px 12px rgba(255,107,107,0.3);
    }
    /* 主楼特殊样式 */
    .directory-item.mainpost { background: linear-gradient(135deg, #4CAF50, #45a049); color: white; }

    /* 状态提示 */
    #directory-loading, #directory-empty {
      text-align: center; padding: 60px 20px; color: rgba(255,255,255,0.6); font-size: 15px;
    }
    #directory-loading::before { content: '📚'; display: block; font-size: 32px; margin-bottom: 10px; animation: pulse 1.5s infinite; }
    #directory-empty::before { content: '📖'; display: block; font-size: 32px; margin-bottom: 10px; opacity: 0.5; }
    @keyframes pulse { 50% { opacity: 0.6; } }

    /* === 遮罩层 === */
    #directory-overlay {
      position: fixed; top: 0; left: 0; width: 100%; height: 100%;
      background: rgba(0,0,0,0.6); z-index: 2147483646;
      opacity: 0; visibility: hidden; transition: 0.3s; backdrop-filter: blur(2px);
    }
    #directory-overlay.show { opacity: 1; visibility: visible; }
`;
  const READER_CSS = `
    /* === 全局基础 === */
    html, body {
    margin: 0;
    padding: 0;
    background: #616161;
    overflow-y: auto;
    color: #fff;
    font-family: "IBM Plex Sans SC", "PingFang SC", "Microsoft YaHei", Arial, sans-serif !important;
    }
    body {
  --img-width: 35vw;
}
body.light-bg {
  background: #f5f5f5 !important;
  color: #222 !important;
}
body.light-bg #cw-title-bar {
  color: #222 !important;
}
   /* === 图片容器 === */
    #cw-container {
      display: flex !important;
      flex-direction: column !important;
      align-items: center !important;
      padding: 20px 0 60px !important;
      gap: 20px !important;
      margin: 0 auto !important;
      will-change: transform;
    }

    #cw-title-bar {
  margin-top: 0;
  margin-bottom: 0;
  margin-left: 0;
  padding-left: 0;
  font-size: 1.25em;
  font-weight: bold;
  font-family: "IBM Plex Sans SC", "PingFang SC", "Microsoft YaHei", Arial, sans-serif !important;
  color: #fff;
  background: none;
  border: none;
  border-radius: 0;
  text-align: left;
  letter-spacing: 0.04em;
  box-shadow: none;
  position: relative;
  z-index: 1;
  user-select: text;
  word-break: break-all;
  width: fit-content;
  max-width: 100vw;
  transition: color 0.3s;
}

    #cw-container img {
      display: block !important;
       width: var(--img-width) !important;
      height: auto !important;
      content-visibility: auto !important;
      contain-intrinsic-size: var(--img-width, 35vw) calc(var(--img-width, 35vw) * 1.5) !important;
      min-height: 650px !important;
      
      transition: transform 0.2s ease, box-shadow 0.2s ease !important;

      background-color: #e0e0e0 !important;
      background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='100%25' height='100%25'%3E%3Ctext x='50%25' y='50%25' dominant-baseline='middle' text-anchor='middle' font-family='sans-serif' font-size='16' fill='%23999999'%3E图片加载中……%3C/text%3E%3C/svg%3E") !important;
      background-position: center center !important;
      background-repeat: no-repeat !important;
      background-size: contain !important;
      border-radius: 0 !important;
      user-select: none !important;
      margin: 0 auto !important;
      box-shadow: 0 4px 8px rgba(0,0,0,0.3);
    }
    #cw-container img.loaded {
      background-image: none !important; 
      background-color: transparent !important;
      min-height: 0 !important;}
    #cw-container img[data-original-size="small"] {
      width: auto !important;
      max-width: var(--img-width, 35vw) !important;
      aspect-ratio: auto !important; /* 小图可能不是漫画页,重置比例 */
    }
    /* === 悬浮控件基础 (退出/工具栏/底部栏) === */
    #cw-exit, #cw-toolbar, #cw-bottom-bar {
        position: fixed; z-index: 2147483647; pointer-events: auto;
    }

    /* 左上退出 */
    #cw-exit {
      top: 15px; left: 20px; background: rgba(0,0,0,0.6); color: #fff; border: none;
      padding: 6px 10px; border-radius: 5px; cursor: pointer; font-size: 14px; backdrop-filter: blur(4px);
    }
      #cw-refresh {
      position: fixed; 
      top: 65px; 
      left: 20px; 
      background: rgba(0,0,0,0.6); 
      color: #fff; 
      border: none;
      padding: 6px 10px; 
      width: 32px;
      text-align: center; 
      border-radius: 5px; 
      cursor: pointer; 
      font-size: 14px; 
      backdrop-filter: blur(4px);
      z-index: 2147483647; 
      transition: 0.2s;
    }
    #cw-refresh:hover { background: rgba(0,0,0,0.8); transform: scale(1.05); }
    /* 右上工具栏 */
    #cw-toolbar { top: 15px; right: 20px; display: flex; flex-direction: column; gap: 10px; }

    /* 底部栏 */
    #cw-bottom-bar {
      bottom: 15px; right: 20px; left: auto !important; width: auto;
      display: flex; align-items: center; gap: 15px;
      background: rgba(0,0,0,0.6); padding: 8px 12px; border-radius: 5px;
      font-size: 14px; backdrop-filter: blur(4px); white-space: nowrap;
      z-index: 2147483650; /* 最高层级 */
    }

    /* 通用按钮样式 */
    #cw-toolbar button, #cw-bottom-bar button {
        color: #fff; border: none; cursor: pointer; transition: 0.2s;
        display: flex; align-items: center; justify-content: center;
    }
    #cw-toolbar button {
        background: rgba(0,0,0,0.6); padding: 6px; width: 32px; height: 32px;
        border-radius: 5px; font-size: 14px; backdrop-filter: blur(4px);
    }
    #cw-bottom-bar button {
        background: rgba(255,255,255,0.2); padding: 4px 8px; border-radius: 3px; font-size: 12px;
    }

    /* 悬停效果 */
    #cw-toolbar button:hover { background: rgba(0,0,0,0.8); transform: scale(1.05); }
    #cw-bottom-bar button:hover { background: rgba(255,255,255,0.3); }

    /* 缩放组 */
    #cw-zoom { display: flex; flex-direction: column; gap: 5px; }
    #cw-zoom-in, #cw-zoom-out { font-size: 20px !important; font-weight: bold !important; }
    #cw-full[data-fullscreen="true"] { font-size: 24px !important; line-height: 1; }
    #cw-page-info { font-weight: bold; color: #fff; }

    /* === 全屏模式 === */
    /* 合并全屏选择器 */
    :is(:fullscreen, :-webkit-full-screen) #cw-toolbar,
    :is(:fullscreen, :-webkit-full-screen) #cw-exit {
        opacity: 0; pointer-events: none; transition: 0.3s;
    }
    :is(:fullscreen, :-webkit-full-screen).tools-visible #cw-toolbar,
    :is(:fullscreen, :-webkit-full-screen).tools-visible #cw-exit {
        opacity: 1; pointer-events: auto;
    }
    /* 全屏下提高 Tooltip 层级 */
    :is(:fullscreen, :-webkit-full-screen) [data-tooltip]:hover::after,
    :is(:fullscreen, :-webkit-full-screen) [data-tooltip]:hover::before {
        z-index: 2147483649 !important;
    }

    /* === Tooltip  === */
    @keyframes tooltipFadeIn { from { opacity: 0; transform: translate(-50%, 5px); } to { opacity: 1; transform: translate(-50%, 0); } }
    @keyframes tooltipFadeInRight { from { opacity: 0; transform: translate(5px, -50%); } to { opacity: 1; transform: translate(0, -50%); } }

    [data-tooltip] { position: relative; overflow: visible !important; }

    /* 共有样式 */
    [data-tooltip]:hover::after {
        content: attr(data-tooltip); position: absolute;
        background: rgba(0,0,0,0.9); color: #fff; padding: 6px 10px;
        border-radius: 4px; font-size: 12px; white-space: nowrap;
        z-index: 2147483648; pointer-events: none; box-shadow: 0 2px 8px rgba(0,0,0,0.3);
        opacity: 0;
    }
    [data-tooltip]:hover::before {
        content: ''; position: absolute; z-index: 2147483648; pointer-events: none; opacity: 0;
        border: 6px solid transparent;
    }

    /* 1. 默认下方 (左上退出按钮) */
    #cw-exit[data-tooltip]:hover::after {
        top: 100%; left: 0; margin-top: 8px; animation: tooltipFadeIn 0.2s ease forwards;
    }
    #cw-exit[data-tooltip]:hover::before {
        top: 100%; left: 15px; margin-top: -4px; border-bottom-color: rgba(0,0,0,0.9); animation: tooltipFadeIn 0.2s ease forwards;
    }

    /* 2. 左侧显示 (右上工具栏) */
    #cw-toolbar [data-tooltip]:hover::after {
        top: 50%; right: 100%; transform: translateY(-50%); margin-right: 10px;
        animation: tooltipFadeInRight 0.2s ease forwards;
    }
    #cw-toolbar [data-tooltip]:hover::before {
        top: 50%; right: 100%; transform: translateY(-50%); margin-right: -2px;
        border-left-color: rgba(0,0,0,0.9); animation: tooltipFadeInRight 0.2s ease forwards;
    }

    /* 3. 上方显示 (底部栏) */
    #cw-bottom-bar [data-tooltip]:hover::after {
        bottom: 100%; left: 50%; transform: translateX(-50%); margin-bottom: 10px;
        animation: tooltipFadeIn 0.2s ease forwards;
    }
    #cw-bottom-bar [data-tooltip]:hover::before {
        bottom: 100%; left: 50%; transform: translateX(-50%); margin-bottom: -2px;
        border-top-color: rgba(0,0,0,0.9); animation: tooltipFadeIn 0.2s ease forwards;
    }
`;

  // 阅读器全局变量定义
  let isReading = false;
  let images = [];
  let bgIsBlack = true;
  let zoomLevel = 35; // 默认缩放比例35%
  let autoReaderEnabled = false; // 全局阅读器开关

  // 目录识别相关变量
  let seriesDirectory = [];
  let currentSeriesKey = "";
  let isDirectoryMode = false;
  let savedThreadTitle = "";
  let originalSeriesTitle = "";
  let searchCache = new Map();
  let directoryMemoryCache = new Map();

  // 保存原始页面HTML
  let readerStartUrl = ""; // 记录进入阅读器时的 URL
  let originalPageHTML = "";
  let originalPageTitle = "";

  // 事件处理与状态
  let readerEventHandlers = {};
  let readerEventsBound = false;
  let readerToolsTimer = null;
  let unbindReaderEvents = null;

  // 缓存前缀与过期时长
  const CACHE_PREFIX = "yamibo-directory-cache-";
  const CACHE_EXPIRY = 24 * 60 * 60 * 1000;

  // 持久化缓存索引键
  const GM_INDEX_KEY = CACHE_PREFIX + "gm-index";

  // 通用存储封装
  function storageGet(key, defaultVal = null) {
    try {
      if (typeof GM_getValue === "function") {
        let val;
        try {
          val = GM_getValue(key, undefined);
        } catch (e) {
          val = undefined;
        }
        if (val === undefined) {
        } else if (val === null) {
          return defaultVal;
        } else if (typeof val === "string") {
          try {
            return JSON.parse(val);
          } catch (e) {
            return val;
          }
        } else {
          return val;
        }
      }
    } catch (e) {}

    try {
      const raw = sessionStorage.getItem(key);
      return raw ? JSON.parse(raw) : defaultVal;
    } catch (e) {
      return defaultVal;
    }
  }

  function storageSet(key, value) {
    try {
      if (typeof GM_setValue === "function") {
        try {
          GM_setValue(key, value);
        } catch (e) {
          try {
            GM_setValue(key, JSON.stringify(value));
          } catch (ee) {
            throw ee;
          }
        }

        try {
          let idx = GM_getValue(GM_INDEX_KEY, undefined);
          if (idx === undefined || idx === null) idx = [];
          if (typeof idx === "string") {
            try {
              idx = JSON.parse(idx);
            } catch (e) {
              idx = [];
            }
          }
          if (!Array.isArray(idx)) idx = [];
          if (!idx.includes(key)) {
            idx.push(key);
            GM_setValue(GM_INDEX_KEY, idx);
          }
        } catch (e) {}
        return true;
      }
    } catch (e) {}

    try {
      sessionStorage.setItem(key, JSON.stringify(value));
      return true;
    } catch (e) {
      console.warn("⚠️ storageSet 失败:", e);
      return false;
    }
  }

  function storageRemove(key) {
    try {
      if (typeof GM_deleteValue === "function") {
        try {
          GM_deleteValue(key);
        } catch (e) {
          try {
            GM_setValue(key, null);
          } catch (ee) {}
        }

        try {
          let idx = GM_getValue(GM_INDEX_KEY, undefined);
          if (typeof idx === "string") {
            try {
              idx = JSON.parse(idx);
            } catch (e) {
              idx = [];
            }
          }
          if (!Array.isArray(idx)) idx = [];
          const newIdx = idx.filter((k) => k !== key);
          GM_setValue(GM_INDEX_KEY, newIdx);
        } catch (e) {}

        return true;
      }
    } catch (e) {}

    try {
      sessionStorage.removeItem(key);
      return true;
    } catch (e) {
      return false;
    }
  }

  // 清理过期缓存:同时支持 sessionStorage 前缀与 GM 索引
  function cleanExpiredCache() {
    let cleanedCount = 0;

    // 1) 清理 sessionStorage 前缀缓存
    try {
      const sKeys = Object.keys(sessionStorage).filter((k) =>
        k.startsWith(CACHE_PREFIX)
      );
      sKeys.forEach((key) => {
        try {
          const raw = sessionStorage.getItem(key);
          if (!raw) return;
          const obj = JSON.parse(raw);
          const ts = obj.ts || obj.timestamp || 0;
          if (Date.now() - ts > CACHE_EXPIRY) {
            sessionStorage.removeItem(key);
            cleanedCount++;
          }
        } catch (e) {
          try {
            sessionStorage.removeItem(key);
            cleanedCount++;
          } catch (ee) {}
        }
      });
    } catch (e) {}

    // 2) 清理 GM_* 存储(使用索引枚举)
    try {
      if (
        typeof GM_getValue === "function" &&
        typeof GM_setValue === "function"
      ) {
        let idx = GM_getValue(GM_INDEX_KEY, undefined);
        if (typeof idx === "string") {
          try {
            idx = JSON.parse(idx);
          } catch (e) {
            idx = [];
          }
        }
        if (!Array.isArray(idx)) idx = [];

        const remaining = [];
        for (const key of idx) {
          try {
            const stored = storageGet(key, null);
            if (!stored) continue;
            const ts = stored.ts || stored.timestamp || 0;
            if (Date.now() - ts > CACHE_EXPIRY) {
              storageRemove(key);
              cleanedCount++;
            } else {
              remaining.push(key);
            }
          } catch (e) {
            try {
              storageRemove(key);
              cleanedCount++;
            } catch (ee) {}
          }
        }

        // 更新索引为剩余项
        try {
          GM_setValue(GM_INDEX_KEY, remaining);
        } catch (e) {}
      }
    } catch (e) {}

    if (cleanedCount > 0)
      console.log("🧹 清理了", cleanedCount, "个过期缓存 (session+GM)");
    return cleanedCount;
  }

  // 在配额不足或需要收缩时清理最旧的缓存(同时支持 sessionStorage 与 GM)
  function cleanOldestCache() {
    try {
      const entries = [];

      // 收集 sessionStorage 条目
      try {
        Object.keys(sessionStorage).forEach((key) => {
          if (!key.startsWith(CACHE_PREFIX)) return;
          try {
            const obj = JSON.parse(sessionStorage.getItem(key));
            const ts = obj.ts || obj.timestamp || 0;
            entries.push({ key, ts, source: "session" });
          } catch (e) {
            entries.push({ key, ts: 0, source: "session" });
          }
        });
      } catch (e) {}

      // 收集 GM_* 条目 via index
      try {
        if (typeof GM_getValue === "function") {
          let idx = GM_getValue(GM_INDEX_KEY, undefined);
          if (typeof idx === "string") {
            try {
              idx = JSON.parse(idx);
            } catch (e) {
              idx = [];
            }
          }
          if (!Array.isArray(idx)) idx = [];

          for (const key of idx) {
            try {
              const stored = storageGet(key, null);
              const ts = stored ? stored.ts || stored.timestamp || 0 : 0;
              entries.push({ key, ts, source: "gm" });
            } catch (e) {
              entries.push({ key, ts: 0, source: "gm" });
            }
          }
        }
      } catch (e) {}

      if (entries.length === 0) return 0;

      // 按时间戳排序,最旧的在前
      entries.sort((a, b) => (a.ts || 0) - (b.ts || 0));

      const toDelete = Math.ceil(entries.length / 2);
      for (let i = 0; i < toDelete; i++) {
        const e = entries[i];
        try {
          if (e.source === "session") sessionStorage.removeItem(e.key);
          else storageRemove(e.key);
        } catch (err) {}
      }

      return toDelete;
    } catch (e) {
      console.warn("cleanOldestCache failed", e);
      return 0;
    }
  }

  // === 全局搜索限流控制器 ===
  // 防止

  const SEARCH_INTERVAL_MS = 12000; // 限制间隔 12 秒 (论坛限制 10 秒,留 2 秒缓冲)
  const MAX_WAIT_TIME = 60000; // 最大等待时间 60 秒,超过则放弃排队

  /**
   * 安全的搜索请求包装器
   * @param {Function} requestFn - 真正执行请求的函数,需要返回 Promise
   * @returns {Promise<any>}
   */
  async function safeSearchRequest(requestFn) {
    let lastTime = 0;
    try {
      if (typeof GM_getValue === "function") {
        lastTime = GM_getValue("last_search_time", 0);
      } else {
        lastTime = parseInt(
          localStorage.getItem("yamibo_last_search_time") || "0"
        );
      }
    } catch (e) {
      lastTime = 0;
    }

    const now = Date.now();
    let waitTime = 0;

    // 计算需要等待的时间
    if (now - lastTime < SEARCH_INTERVAL_MS) {
      waitTime = SEARCH_INTERVAL_MS - (now - lastTime);
      // 添加随机抖动
      waitTime += Math.random() * 1000;
    }

    // 异常等待时间重置
    if (waitTime > MAX_WAIT_TIME) {
      console.warn(`[限流] 等待时间过长 (${waitTime}ms),重置计时器`);
      waitTime = 1000;
    }

    // 执行等待 (静默模式)
    if (waitTime > 0) {
      console.log(
        `[限流] 搜索请求过于频繁,后台静默排队 ${(waitTime / 1000).toFixed(
          1
        )} 秒...`
      );
      await new Promise((resolve) => setTimeout(resolve, waitTime));
    }

    // 更新全局时间戳 (抢锁)
    try {
      const newNow = Date.now();
      if (typeof GM_setValue === "function") {
        GM_setValue("last_search_time", newNow);
      } else {
        localStorage.setItem("yamibo_last_search_time", newNow.toString());
      }
    } catch (e) {
      console.error("[限流] 更新时间戳失败", e);
    }

    // 执行真正的请求
    return await requestFn();
  }

  // 检查是否启用了全局阅读器模式
  function checkAutoReaderStatus() {
    const stored = localStorage.getItem("yamibo-auto-reader");
    return stored === "true";
  }

  // 设置全局阅读器模式
  function setAutoReaderStatus(enabled) {
    autoReaderEnabled = enabled;
    localStorage.setItem("yamibo-auto-reader", enabled.toString());
    console.log("🔧 全局阅读器模式:", enabled ? "开启" : "关闭");
  }

  /*** 标题标准化函数  ***/
  function normalizeSeriesTitle(rawTitle) {
    if (!rawTitle) return "";

    // 首先清理页面title的后缀
    let title = rawTitle;
    if (title.includes(" - ")) {
      title = title.split(" - ")[0].trim();
    }

    // 移除页数标注
    title = title.replace(/[((]\s*\d+\s*p\s*[))]\s*$/i, "");

    // 移除章节标识 - 参考脚本的模式
    title = title.replace(
      /第[\d一二三四五六七八九十百千万〇零兩两1234567890\.]+[话話章节節回卷篇]/gi,
      " "
    );
    title = title.replace(/\d+(?:\.\d+)?\s*[话話章节節回卷篇]/gi, " ");
    title = title.replace(
      /\d+(?:\.\d+)?\s*[上下前后前後左右中篇部卷期全完]+(?:\s*[++&和及與并並,,/]\s*[上下前后前後左右中篇部卷期全完]+)+/gi,
      " "
    );
    title = title.replace(
      /[((][上下前后前後中全完]+(?:\s*[,,++&和及與并並/]\s*[上下前后前後中全完]+)*[))]/g,
      " "
    );
    title = title.replace(
      /\d+(?:\.\d+)?\s*[上下前后前後左右中篇部卷期全完]+/gi,
      " "
    );
    title = title.replace(
      /(?:\s+|[-‐‑‒–—―-~~·•_、::])?\d+(?:\.\d+)*(?:\s*[上下前后前後左右中篇部卷期話话节節全完])?\s*$/g,
      " "
    );
    title = title.replace(/[--—–~~\u2013\u2014\s]+$/g, " ");
    title = title.replace(/[\[\]【】()()]/g, " ");
    title = title.replace(/\s+/g, " ").trim();

    if (!title) {
      return rawTitle.trim();
    }
    return title;
  }

  function buildSeriesKey(title) {
    const normalized = normalizeSeriesTitle(title || "");
    const base = normalized || (title || "").trim();
    return base.toLowerCase();
  }

  function generateSeriesKey(title) {
    try {
      const k = buildSeriesKey(title || "");
      // 限制长度并移除特殊字符,避免存储键过长或包含非法字符
      return k.replace(/[\s\/:\\#\?&=\+%\*\|<>"'`]/g, "-").substring(0, 120);
    } catch (e) {
      return String(title || "")
        .toLowerCase()
        .substring(0, 120);
    }
  }

  // 统一的排除词汇列表,避免重复定义
  const EXCLUDE_WORDS = [
    "个人汉化",
    "汉化组",
    "汉化",
    "个人翻译",
    "個人翻譯",
    "个汉",
    "翻译",
    "生肉",
    "未翻译",
    "填坑组",
    "粮食组",
    "保护协会",
    "創作百合",
    "创作百合",
    "熟肉",
    "字幕",
    "工作室",
    "提灯喵",
    "社团",
    "官方中字",
    "出版",
    "汉化工房",
    "猫岛汉化组",
    "和菓子漫画屋",
    "大友同好會",
    "透明声彩汉化组",
    "翻译组",
    "汉化委员会",
    "渣翻渣嵌",
    "合作汉化",
    "完结",
    "连载",
    "短篇",
    "合集",
    "同人",
    "raw",
    "韩漫",
    "杂志",
    "英译",
    "单行本",
    "插画",
    "特典",
    "番外",
    "外传",
    "彩页",
    "合订本",
  ].map((s) => s.toLowerCase());

  // 统一的文本过滤函数:检查是否为噪音词汇
  function isNoiseText(text) {
    if (!text || typeof text !== "string") return true;
    // [优化] 直接使用预编译正则,替代循环 includes
    if (EXCLUDE_WORDS_REGEX.test(text)) return true;

    // 常见格式判断:纯数字视为噪音
    if (/^\d+$/.test(text)) return true;

    // 智能长度检查
    const hasCJK =
      /[\u4e00-\u9fa5\u3040-\u309F\u30A0-\u30FF\uAC00-\uD7AF]/.test(text);
    const minLength = hasCJK ? 2 : 3;
    if (text.length < minLength) return true;

    return false;
  }

  // 保持向后兼容的 isGenericTag 函数
  function isGenericTag(text) {
    return isNoiseText(text);
  }

  function getCachedDirectory(seriesKey) {
    // 🚫 临时禁用缓存读取,强制测试双语解析逻辑
    console.log("🔄 缓存读取已禁用,将执行新的双语解析逻辑");
    return null;

    if (!seriesKey) return null;

    // 1) 先检查内存缓存
    try {
      if (directoryMemoryCache && directoryMemoryCache.has(seriesKey)) {
        const cached = directoryMemoryCache.get(seriesKey);
        if (cached && cached.ts && Date.now() - cached.ts < CACHE_EXPIRY) {
          return cached.data || cached.d || null;
        } else {
          directoryMemoryCache.delete(seriesKey);
        }
      }
    } catch (e) {}

    const key = CACHE_PREFIX + seriesKey;

    // 2) 尝试 GM_getValue
    try {
      if (typeof GM_getValue === "function") {
        let stored = GM_getValue(key, undefined);
        if (typeof stored === "string") {
          try {
            stored = JSON.parse(stored);
          } catch (e) {
            /* leave as-is */
          }
        }
        if (stored && (stored.d || stored.data)) {
          const ts = stored.ts || stored.timestamp || 0;
          if (Date.now() - ts < CACHE_EXPIRY) {
            try {
              directoryMemoryCache.set(seriesKey, {
                d: stored.d || stored.data,
                data: stored.d || stored.data,
                ts: ts || Date.now(),
              });
            } catch (e) {}
            return stored.d || stored.data;
          } else {
            try {
              storageRemove(key);
            } catch (e) {}
            return null;
          }
        }
      }
    } catch (e) {}

    // 3) 尝试 sessionStorage
    try {
      const raw = sessionStorage.getItem(key);
      if (raw) {
        let obj = raw;
        try {
          obj = JSON.parse(raw);
        } catch (e) {
          obj = raw;
        }
        if (obj && (obj.d || obj.data)) {
          const ts = obj.ts || obj.timestamp || 0;
          if (Date.now() - ts < CACHE_EXPIRY) {
            try {
              directoryMemoryCache.set(seriesKey, {
                d: obj.d || obj.data,
                data: obj.d || obj.data,
                ts: ts || Date.now(),
              });
            } catch (e) {}
            return obj.d || obj.data;
          } else {
            try {
              sessionStorage.removeItem(key);
            } catch (e) {}
            return null;
          }
        }
      }
    } catch (e) {}

    return null;
  }

  function setCachedDirectory(seriesKey, directory) {
    if (!seriesKey || !directory) return false;
    const key = CACHE_PREFIX + seriesKey;
    const payload = { ts: Date.now(), data: directory, d: directory };

    try {
      directoryMemoryCache.set(seriesKey, {
        d: directory,
        data: directory,
        ts: payload.ts,
      });
    } catch (e) {}

    try {
      if (typeof GM_setValue === "function") {
        try {
          GM_setValue(key, payload);
        } catch (e) {
          try {
            GM_setValue(key, JSON.stringify(payload));
          } catch (ee) {
            /* ignore */
          }
        }
        return true;
      }
    } catch (e) {}

    try {
      sessionStorage.setItem(key, JSON.stringify(payload));
      return true;
    } catch (e) {
      console.warn("⚠️ setCachedDirectory 写入失败:", e);
      return false;
    }
  }

  /*** 相似度计算辅助函数  ***/

  // 简繁体标准化函数 - 使用 OpenCC 库
  let converter = null;

  function normalizeChineseVariants(text) {
    try {
      // 初始化转换器(繁体转简体)
      if (!converter && typeof OpenCC !== "undefined") {
        converter = OpenCC.Converter({ from: "tw", to: "cn" });
      }

      if (converter) {
        return converter(text);
      }

      return text;
    } catch (error) {
      console.warn("简繁转换失败,使用原文本:", error);
      return text;
    }
  }
  //中外语种统一判断
  function isForeignText(str) {
    return (
      /[A-Za-z\u3040-\u30FF\uAC00-\uD7AF]/.test(str) &&
      !/[\u4e00-\u9fa5]/.test(str)
    );
  }
  // 提取有意义的关键词(排除常见无关词汇)
  function extractMeaningfulKeywords(title) {
    const core = extractCoreWorkName(title);
    if (typeof core === "object" && core._dualLanguage) {
      return [core._dualLanguage.chinese, core._dualLanguage.foreign].filter(
        Boolean
      );
    }
    if (Array.isArray(core)) {
      return core.filter(Boolean);
    }
    return [core.toString().trim()];
  }
  // 生成搜索关键词
  function generateSearchTerms(seriesTitle) {
    if (!seriesTitle) return [];
    const core = extractCoreWorkName(seriesTitle);
    if (typeof core === "object" && core._dualLanguage) {
      return [core._dualLanguage.foreign, core._dualLanguage.chinese].filter(
        Boolean
      );
    }
    if (Array.isArray(core)) {
      return core.filter(Boolean);
    }
    return [core.toString().trim()];
  }

  // 计算核心作品名相似度
  function calculateCoreWorkSimilarity(core1, core2) {
    const arr1 = Array.isArray(core1) ? core1 : [core1];
    const arr2 = Array.isArray(core2) ? core2 : [core2];
    let maxScore = 0;
    for (const s1 of arr1) {
      for (const s2 of arr2) {
        const a = s1.toString().trim().toLowerCase();
        const b = s2.toString().trim().toLowerCase();
        if (!a || !b) continue;
        if (a === b) return 1.0;
        if (a.includes(b) || b.includes(a)) {
          const longer = a.length > b.length ? a : b;
          const shorter = a.length <= b.length ? a : b;
          maxScore = Math.max(
            maxScore,
            (shorter.length / longer.length) * 0.95
          );
        }
        // 最长公共子串
        let maxCommonLength = 0;
        for (let i = 0; i < a.length; i++) {
          for (let j = 0; j < b.length; j++) {
            let commonLength = 0;
            while (
              i + commonLength < a.length &&
              j + commonLength < b.length &&
              a[i + commonLength] === b[j + commonLength]
            ) {
              commonLength++;
            }
            maxCommonLength = Math.max(maxCommonLength, commonLength);
          }
        }
        const minLength = Math.min(a.length, b.length);
        if (maxCommonLength >= 4) {
          const similarity = (maxCommonLength / minLength) * 0.9;
          maxScore = Math.max(maxScore, similarity);
        }
      }
    }
    return maxScore;
  }

  // 检查关键词匹配(排除翻译组等无关信息)
  function checkKeywordMatch(core1, core2) {
    const kws1 = extractMeaningfulKeywords(core1);
    const kws2 = extractMeaningfulKeywords(core2);
    let bestMatch = 0;
    for (const kw1 of kws1) {
      for (const kw2 of kws2) {
        if (kw1.length >= 2 && kw2.length >= 2) {
          if (kw1 === kw2) {
            return 0.95;
          } else if (kw1.includes(kw2) || kw2.includes(kw1)) {
            const longer = kw1.length > kw2.length ? kw1 : kw2;
            const shorter = kw1.length <= kw2.length ? kw1 : kw2;
            const match = shorter.length / longer.length;
            bestMatch = Math.max(bestMatch, match * 0.9);
          }
        }
      }
    }
    return bestMatch;
  }

  // 提取核心作品名
  function extractCoreWorkName(title) {
    console.log(`   🔍 重构版标题处理: "${title}"`);
    if (!title || typeof title !== "string") return "";

    // === 0. 引用全局噪音 ===

    function getScriptType(str) {
      const hasKana = /[\u3040-\u30FF]/.test(str);
      const hasHanzi = /[\u4e00-\u9fa5]/.test(str);
      const hasLatin = /[a-zA-Z]/.test(str);
      const hasKorean = /[\uAC00-\uD7AF]/.test(str);
      if (hasKorean) return "KR";
      if (hasKana) return "JP";
      if (hasHanzi && !hasKana) return "ZH";
      if (hasLatin) return "EN";
      return "OTHER";
    }

    function isFatalNoise(text) {
      return FATAL_KEYWORDS_REGEX.test(text);
    }

    // === 1. 去头 ===
    let body = title.trim();
    while (
      body.match(/^(\s*【[^】]*】\s*)+/) ||
      body.match(/^(\s*\[[^\]]*\]\s*)+/) ||
      body.match(/^(\s*[\[【((][^\]】))]*[\]】))]\s*)+/)
    ) {
      body = body
        .replace(/^(\s*【[^】]*】\s*)+/g, "")
        .replace(/^(\s*\[[^\]]*\]\s*)+/g, "")
        .replace(/^(\s*[\[【((][^\]】))]*[\]】))]\s*)+/g, "")
        .replace(/^[\]】))]\s*/, "")
        .trim();
    }
    if (!body) body = title.trim();

    // === 1.5 枢轴切割 ===
    const fatalSource = FATAL_KEYWORDS_REGEX.source;
    const pivotRegex = new RegExp(
      `[【\\[][^\\]】]*(?:${fatalSource})[^\\]】]*[】\\]]`,
      "gi"
    );
    const pivotParts = body.split(pivotRegex);

    if (pivotParts.length > 1) {
      const lastSegment = pivotParts[pivotParts.length - 1].trim();
      // 检查最后一段是否包含有效字符 (字母、数字、汉字),防止只剩下 " / " 这种符号
      if (/[a-zA-Z0-9\u4e00-\u9fa5]/.test(lastSegment)) {
        console.log(
          `   ✂️ Step 1.5 枢轴切割: 丢弃左侧元数据,保留 "${lastSegment}"`
        );
        body = lastSegment;
      }
    }

    // === 2. 断尾 ===
    let coreCandidate = body;

    // [修正] 使用 {1,20} 替代 +,避免 Nothing to repeat 错误
    const explicitRe = new RegExp(
      `(` +
        `(?:最终|完结|后日谈|前日谈|特别|番外|短篇|尾声|序章|终章)[话話章节節篇]?|` +
        `第?${NUM_RANGE_UNIT}{1,20}[话話章节節回卷篇期部]|` +
        `(?:Vol|Part|Ch|Ep|Ex|番外|篇)\\.?\\s*${NUM_RANGE_UNIT}{1,20}|` +
        `[##]${NUM_RANGE_UNIT}{1,20}` +
        `)`,
      "i"
    );

    const spaceNumRe = new RegExp(
      `\\s+${NUM_RANGE_UNIT}{1,20}(?:[..]\\d+)?(?:\\s+|$)`
    );
    const stickyNumRe = new RegExp(
      `([^a-zA-Z0-9\\s])(${NUM_CHAR}{1,4}(?:[..]\\d+)?)(?:\\s+|$)`
    );

    const mExplicit = body.match(explicitRe);
    const mSpace = body.match(spaceNumRe);
    const mSticky = body.match(stickyNumRe);

    let splitIndex = body.length;
    let splitType = "";

    if (mExplicit && mExplicit.index < splitIndex) {
      splitIndex = mExplicit.index;
      splitType = "显式章节";
    }
    if (mSpace && mSpace.index < splitIndex) {
      splitIndex = mSpace.index;
      splitType = "隐式空格数字";
    }
    if (mSticky) {
      const stickyCutIndex = mSticky.index + mSticky[1].length;
      if (stickyCutIndex < splitIndex) {
        splitIndex = stickyCutIndex;
        splitType = "粘连数字";
      }
    }

    // 后方保护
    if (splitIndex < body.length) {
      const remainingPart = body.substring(splitIndex);
      if (/[\/||/]/.test(remainingPart)) {
        console.log(`   🛡️ Step 2 触发后方保护: 检测到分隔符,跳过断尾`);
      } else {
        const potentialTitle = body.substring(0, splitIndex).trim();
        if (potentialTitle.length >= 1) {
          console.log(
            `   🗡️ Step 2 ${splitType}切割: 保留左侧 "${potentialTitle}"`
          );
          coreCandidate = potentialTitle;
        }
      }
    }

    // === 3. 分词 ===
    coreCandidate = coreCandidate.replace(/\s+[-_—–]\s+/g, "/");
    coreCandidate = coreCandidate.replace(
      /([^\x00-\x7F])\s*[-_—–]\s*([^\x00-\x7F])/g,
      "$1/$2"
    );
    coreCandidate = coreCandidate.replace(
      /([^\x00-\x7F])\s*[-_—–]\s*([a-zA-Z0-9])/g,
      "$1/$2"
    );
    coreCandidate = coreCandidate.replace(
      /([a-zA-Z0-9])\s*[-_—–]\s*([^\x00-\x7F])/g,
      "$1/$2"
    );

    const SPLIT_RE = /[\/||/()()\[\]【】]+/;
    let rawParts = coreCandidate
      .split(SPLIT_RE)
      .map((p) => p.trim())
      .filter(Boolean);

    // === 4. 清洗 ===
    function cleanTitlePart(str) {
      if (!str) return "";
      let s = str.trim();
      if (isFatalNoise(s)) return "";

      s = s.replace(/「[^」]*」/g, "");
      s = s.replace(/『[^』]*』/g, "");
      s = s.replace(/C\d{2,3}/gi, "");
      // [优化] 使用全局正则替换噪音词
      s = s.replace(EXCLUDE_WORDS_REGEX, "");

      // 移除章节号 [修正正则]
      s = s.replace(
        new RegExp(
          `第?${NUM_RANGE_UNIT}{1,20}[话話章节節回卷篇期部](?:[\\s\\d]+)?`,
          "g"
        ),
        ""
      );
      s = s.replace(new RegExp(`其${NUM_RANGE_UNIT}+`, "g"), "");
      s = s.replace(
        /(?:最终|完结|后日谈|前日谈|特别|番外|短篇|尾声|序章|终章)[话話章节節篇]?/g,
        ""
      );
      s = s.replace(
        new RegExp(
          `(?:Vol|Part|Ch|Ep|Ex|番外|篇)\\.?\\s*${NUM_RANGE_UNIT}{1,20}`,
          "ig"
        ),
        ""
      );
      s = s.replace(new RegExp(`[##]${NUM_RANGE_UNIT}{1,20}`, "g"), "");

      s = s.trim();
      // 移除末尾版本号/数字
      s = s.replace(/v\d+(\.\d+)?/gi, "");
      s = s.replace(
        new RegExp(`\\s+${NUM_RANGE_UNIT}{1,20}(\\.\\d+)?$`, "g"),
        ""
      );
      s = s.replace(
        new RegExp(`(${NUM_RANGE_UNIT}{1,20}(\\.\\d+)?)$`, "g"),
        ""
      );
      s = s.replace(/[①-⑳]$/, "");
      s = s.replace(/[-~—–]+$/g, "");

      return s.trim();
    }

    let validParts = [];
    let seen = new Set();

    for (const p of rawParts) {
      const cleaned = cleanTitlePart(p);
      const isCJK = /[\u4e00-\u9fa5\u3040-\u30FF]/.test(cleaned);
      if (cleaned && !/^\d+$/.test(cleaned) && !seen.has(cleaned)) {
        if ((isCJK && cleaned.length >= 2) || (!isCJK && cleaned.length >= 3)) {
          validParts.push(cleaned);
          seen.add(cleaned);
        }
      }
    }
    console.log(`   🧺 Step 3 清洗后候选词: ${JSON.stringify(validParts)}`);

    // === 5. 输出 ===
    const zhParts = [];
    const foreignParts = [];

    for (const part of validParts) {
      const type = getScriptType(part);
      if (type === "ZH") zhParts.push(part);
      else if (type === "JP" || type === "EN" || type === "KR")
        foreignParts.push(part);
      else {
        if (/[\u4e00-\u9fa5]/.test(part)) zhParts.push(part);
        else foreignParts.push(part);
      }
    }

    if (zhParts.length > 0 && foreignParts.length > 0) {
      const bestChinese = zhParts.sort((a, b) => b.length - a.length)[0];
      const bestForeign = foreignParts[0];
      const result = new String(bestChinese);
      result._dualLanguage = { chinese: bestChinese, foreign: bestForeign };
      return result;
    }

    if (zhParts.length > 0) {
      if (zhParts.length >= 2) {
        const result = zhParts;
        result._multiChinese = true;
        return result;
      }
      return zhParts[0];
    }

    if (foreignParts.length > 0) {
      if (foreignParts.length >= 2) {
        if (foreignParts[1].includes(foreignParts[0])) return foreignParts[1];
        if (foreignParts[0].includes(foreignParts[1])) return foreignParts[0];
        const result = [foreignParts[0], foreignParts[1]];
        result._multiForeign = true;
        return result;
      }
      return foreignParts[0];
    }

    if (validParts.length === 0) {
      return coreCandidate || body;
    }

    return validParts[0];
  }

  // 智能相似度匹配函数 - 专注核心作品名匹配
  function calculateSimilarity(searchTitle, resultTitle) {
    console.log(`🔄 计算相似度: "${searchTitle}" vs "${resultTitle}"`);

    // 提取两个标题的核心作品名(去除翻译组、章节号等干扰信息)
    const searchCore = extractCoreWorkName(searchTitle);
    const resultCore = extractCoreWorkName(resultTitle);

    console.log(`   📝 搜索核心: "${searchCore}"`);
    console.log(`   📝 结果核心: "${resultCore}"`);

    // 调试:检查是否有双语信息
    if (typeof searchCore === "object" && searchCore._dualLanguage) {
      console.log(
        `   🌐 搜索核心双语信息: 中文="${searchCore._dualLanguage.chinese}" 外文="${searchCore._dualLanguage.foreign}"`
      );
    }
    if (typeof resultCore === "object" && resultCore._dualLanguage) {
      console.log(
        `   🌐 结果核心双语信息: 中文="${resultCore._dualLanguage.chinese}" 外文="${resultCore._dualLanguage.foreign}"`
      );
    }

    // 智能长度检查:CJK字符2个即有效,英文需要3个
    function isValidCore(core) {
      if (!core) return false;
      const coreStr = core.toString();
      const hasCJK =
        /[\u4e00-\u9fa5\u3040-\u309F\u30A0-\u30FF\uAC00-\uD7AF]/.test(coreStr);
      return hasCJK ? coreStr.length >= 2 : coreStr.length >= 3;
    }

    if (!isValidCore(searchCore) || !isValidCore(resultCore)) {
      console.log(`   ❌ 核心标题无效,跳过`);
      return 0;
    }

    // 检查是否有关键词直接匹配(排除翻译组等无关词汇)
    const keywordMatch = checkKeywordMatch(searchCore, resultCore);
    if (keywordMatch > 0) {
      console.log(`   ✅ 关键词匹配度: ${(keywordMatch * 100).toFixed(1)}%`);
      return keywordMatch;
    }

    // 计算核心作品名的相似度
    const similarity = calculateCoreWorkSimilarity(searchCore, resultCore);
    console.log(`   ✅ 核心作品相似度: ${(similarity * 100).toFixed(1)}%`);

    return similarity;
  }

  /*** 目录识别功能 ***/

  /*** 搜索流程 ***/
  // 解析搜索结果页面 HTML,返回原始结果列表
  function parseSearchHtml(html) {
    const parser = new DOMParser();
    const doc = parser.parseFromString(html, "text/html");
    const results = [];
    const seenUrls = new Set();

    // 兼容多种选择器,覆盖不同主题和页面结构
    const selectors = [
      "#threadlist li.pbw h3.xs3 a",
      "#threadlist ul li.pbw h3.xs3 a",
      "#threadlist .slst ul li.pbw h3 a",
      "#threadlist li h3.xs3 a",
      "#threadlist .xst",
      "a.xst",
      "#threadlist tbody .subject a",
      ".threadlist .title a",
      'tbody[id^="normalthread"] .xst',
    ];

    selectors.forEach((selector) => {
      const elements = doc.querySelectorAll(selector);
      elements.forEach((link) => {
        let url = link.href;
        const title = link.textContent.trim();

        // 基础过滤
        if (!url || !title || title === "快速") return;

        // URL 补全
        if (url.startsWith("/")) {
          url = `https://bbs.yamibo.com${url}`;
        } else if (url.startsWith("forum.php")) {
          url = `https://bbs.yamibo.com/${url}`;
        }

        // 提取 Thread ID
        const threadId =
          url.match(/tid=(\d+)/)?.[1] || url.match(/thread-(\d+)-/)?.[1];
        if (!threadId) return;

        // 去重
        if (!seenUrls.has(threadId)) {
          seenUrls.add(threadId);
          results.push({
            threadId,
            title, // 原始标题
            url,
          });
        }
      });
    });

    // 检查是否有下一页
    const hasNextPage = !!doc.querySelector('.nxt, .next, a[href*="page=2"]');

    return { results, hasNextPage };
  }
  // 执行单次关键词搜索 (修正翻页逻辑)
  async function fetchSearchResults(searchTerm, forumId) {
    console.log(`🔍 执行搜索: "${searchTerm}" (fid=${forumId})`);
    const allResults = [];

    // 1. 获取 FormHash
    let formHash = getFormHash();
    if (!formHash) {
      try {
        const preRes = await fetch("https://bbs.yamibo.com/search.php");
        const preHtml = await preRes.text();
        const m = preHtml.match(/name="formhash"\s+value="([^"]+)"/);
        if (m) formHash = m[1];
      } catch (e) {}
    }

    // 2. 准备第一页请求 (这算一次搜索,受限流控制)

    const performFirstPage = async () => {
      const encodedTerm = encodeURIComponent(searchTerm);
      // 尝试 GET 搜索
      const getUrl = `https://bbs.yamibo.com/search.php?mod=forum&srchtxt=${encodedTerm}&formhash=${formHash}&srchfid=${forumId}&orderby=dateline&ascdesc=desc&searchsubmit=yes`;

      let response = await fetch(getUrl, {
        method: "GET",
        credentials: "include",
        headers: { "Cache-Control": "no-cache" },
      });

      // GET 失败尝试 POST
      if (!response.ok) {
        const formData = new FormData();
        formData.append("formhash", formHash);
        formData.append("srchtxt", searchTerm);
        formData.append("srchfid[]", forumId);
        formData.append("orderby", "dateline");
        formData.append("ascdesc", "desc");
        formData.append("searchsubmit", "yes");

        response = await fetch("https://bbs.yamibo.com/search.php?mod=forum", {
          method: "POST",
          credentials: "include",
          body: formData,
        });
      }
      return response;
    };

    try {
      // === 第1页搜索 (消耗限流配额) ===
      const response = await safeSearchRequest(performFirstPage);

      if (!response.ok) {
        console.warn(`❌ 搜索请求失败: ${response.status}`);
        return [];
      }

      // **关键点:获取最终 URL 以提取 searchid**
      // fetch 会自动跟随重定向,response.url 就是最终的 searchid URL
      const finalUrl = response.url;
      console.log(`   🔗 搜索结果URL: ${finalUrl}`);

      const html = await response.text();
      if (html.includes("需要登录")) return [];

      const { results, hasNextPage } = parseSearchHtml(html);
      allResults.push(...results);
      console.log(`   📄 第1页找到 ${results.length} 个结果`);

      // === 第2页搜索 (利用 searchid 翻页,不消耗限流配额) ===
      if (hasNextPage) {
        const searchIdMatch = finalUrl.match(/searchid=(\d+)/);

        if (searchIdMatch) {
          const searchId = searchIdMatch[1];
          console.log(
            `   📄 检测到第2页,使用 searchid=${searchId} 快速翻页 (不触发限流)`
          );

          // 构造纯翻页链接
          const page2Url = `https://bbs.yamibo.com/search.php?mod=forum&searchid=${searchId}&orderby=dateline&ascdesc=desc&searchsubmit=yes`;

          try {
            const page2Res = await fetch(page2Url, {
              method: "GET",
              credentials: "include",
            });

            if (page2Res.ok) {
              const page2Html = await page2Res.text();
              const page2Data = parseSearchHtml(page2Html);
              allResults.push(...page2Data.results);
              console.log(`   📄 第2页找到 ${page2Data.results.length} 个结果`);
            }
          } catch (e) {
            console.warn("⚠️ 快速翻页失败:", e);
          }
        } else {
          console.warn("⚠️ 未能提取 searchid,跳过第2页以避免限流");
        }
      }
    } catch (error) {
      console.error(`❌ 搜索词 "${searchTerm}" 执行出错:`, error);
    }

    return allResults;
  }
  // 主搜索流程
  async function searchSeriesDirectory(seriesTitle) {
    if (!seriesTitle) return [];

    // 1. 检查内存缓存
    if (searchCache.has(seriesTitle)) {
      console.log("📦 命中内存缓存");
      return searchCache.get(seriesTitle);
    }

    // 2. 检查持久化缓存
    const seriesKey = generateSeriesKey(seriesTitle);
    const cached = getCachedDirectory(seriesKey);
    if (cached && cached.length > 0) {
      console.log("📦 命中持久化缓存");
      searchCache.set(seriesTitle, cached);
      return cached;
    }

    console.log("🔍 开始搜索系列目录:", seriesTitle);
    const forumId = detectForumId();

    // 3. 生成关键词列表
    // 直接使用 extractCoreWorkName 的结果,不做多余的双语判断
    const coreName = extractCoreWorkName(seriesTitle);
    let searchTerms = [];

    if (typeof coreName === "object" && coreName._dualLanguage) {
      // 双语对象:分别搜中文和外文
      searchTerms = [
        coreName._dualLanguage.chinese,
        coreName._dualLanguage.foreign,
      ];
    } else if (Array.isArray(coreName)) {
      // 多中文别名:全部加入
      searchTerms = coreName;
    } else {
      // 单语:直接搜
      searchTerms = [coreName.toString()];
    }

    // 过滤空词和重复词
    searchTerms = [...new Set(searchTerms)].filter((t) => t && t.length >= 2);
    console.log("🎯 生成搜索关键词:", searchTerms);

    // 4. 执行搜索并合并结果
    // 使用 Map 按 threadId 去重,保留相似度高的
    const resultMap = new Map();

    // 限制最多搜前2个词 (避免请求过多)
    const termsToSearch = searchTerms.slice(0, 2);

    for (const term of termsToSearch) {
      const results = await fetchSearchResults(term, forumId);

      results.forEach((item) => {
        // 计算相似度
        const similarity = calculateSimilarity(seriesTitle, item.title);
        console.log(
          `   [匹配] "${item.title}" 相似度: ${(similarity * 100).toFixed(1)}%`
        );

        // 只有相似度达标才保留 (0.7 阈值)
        if (similarity > 0.7) {
          const existing = resultMap.get(item.threadId);
          // 如果是新结果,或者新结果相似度更高,则更新
          if (!existing || similarity > existing.similarity) {
            resultMap.set(item.threadId, {
              ...item,
              originalTitle: item.title, // 保持字段兼容
              normalizedTitle: normalizeSeriesTitle(item.title),
              similarity,
              searchTerm: term,
            });
          }
        }
      });
    }

    // 5. 结果排序与转换
    let directory = Array.from(resultMap.values());

    // 按 Thread ID 升序 (老帖子在前,新帖子在后,符合漫画章节发布顺序)
    directory.sort((a, b) => parseInt(a.threadId) - parseInt(b.threadId));

    console.log(`📊 最终筛选结果: ${directory.length} 个章节`);

    // 6. 兜底策略:如果没搜到,返回当前页作为单章节目录
    if (directory.length === 0) {
      const currentTid = getCurrentThreadId();
      if (currentTid) {
        console.log("🔧 搜索无果,使用当前章节兜底");
        return [
          {
            threadId: currentTid,
            title: seriesTitle,
            originalTitle: seriesTitle,
            url: window.location.href,
            normalizedTitle: normalizeSeriesTitle(seriesTitle),
            similarity: 1.0,
            searchTerm: "current",
          },
        ];
      }
    } else {
      // 写入缓存
      searchCache.set(seriesTitle, directory);
      setCachedDirectory(seriesKey, directory);
    }

    return directory;
  }

  // 提取章节号(支持小数格式和复杂排序)
  function extractChapterNumber(title) {
    if (!title || typeof title !== "string") return 0;
    // 特殊章节(番外/彩页等)放到最后
    const specialPatterns = [
      /番外|外传|特别篇|SP|特典|彩页|后记|前记|预告|PV|CM/i,
      /extra|special|omake|bonus/i,
    ];
    for (const p of specialPatterns) {
      if (p.test(title)) return 99999;
    }

    const patterns = [
      {
        regex: /(?:第\s*)?(\d+\.\d+)\s*[话話章节節回卷篇期]?/i,
        priority: 1,
        type: "decimal",
      },
      {
        regex: /第\s*(\d+)\s*[话話章节節回卷篇]/i,
        priority: 2,
        type: "standard",
      },
      { regex: /(\d+)\s*[话話章节節]/i, priority: 3, type: "numbered" },
      {
        regex: /[\[\((](\d+(?:\.\d+)?)[\]\))]/,
        priority: 4,
        type: "bracketed",
      },
      { regex: /(?:第\s*)?(\d+)\s*卷/i, priority: 5, type: "volume" },
      { regex: /\s(\d+(?:\.\d+)?)$/, priority: 6, type: "trailing" },
      { regex: /^(\d+(?:\.\d+)?)\s/, priority: 7, type: "leading" },
      { regex: /(\d+(?:\.\d+)?)/, priority: 8, type: "any" },
    ];

    let bestMatch = null;
    let bestPriority = Infinity;
    for (const p of patterns) {
      const m = title.match(p.regex);
      if (m && p.priority < bestPriority) {
        bestMatch = m[1];
        bestPriority = p.priority;
      }
    }

    if (bestMatch != null) {
      if (bestMatch.includes(".")) {
        // 小数章如 9.5 -> 9500(保持整数比较)
        const v = parseFloat(bestMatch);
        if (!isNaN(v)) return Math.round(v * 1000);
      } else {
        const v = parseInt(bestMatch, 10);
        if (!isNaN(v)) return v;
      }
    }

    // 简单中文数字映射(可扩展)
    const chineseNumbers = {
      零: 0,
      一: 1,
      二: 2,
      三: 3,
      四: 4,
      五: 5,
      六: 6,
      七: 7,
      八: 8,
      九: 9,
      十: 10,
      十一: 11,
      十二: 12,
      十三: 13,
      十四: 14,
      十五: 15,
      十六: 16,
      十七: 17,
      十八: 18,
      十九: 19,
      二十: 20,
    };
    for (const [ch, num] of Object.entries(chineseNumbers)) {
      if (
        title.includes(`第${ch}`) ||
        title.includes(`${ch}话`) ||
        title.includes(`${ch}章`)
      ) {
        return num;
      }
    }

    return 0;
  }

  function getFormHash() {
    if (isReading) {
      console.log("🔑 FormHash: 阅读模式下跳过FormHash");
      return "";
    }
    const formHashInput = document.querySelector('input[name="formhash"]');
    return formHashInput ? formHashInput.value : "";
  }

  // 从URL识别帖子所属版块
  function detectForumId(url = window.location.href) {
    // 方式1: 从URL参数中提取 fid
    const fidMatch = url.match(/[?&]fid=(\d+)/);
    if (fidMatch) {
      const fid = fidMatch[1];
      console.log(`📍 从URL参数识别版块: fid=${fid}`);
      return fid;
    }

    // 方式2: 从forum.php路径中提取
    const forumMatch = url.match(/forum-(\d+)-/);
    if (forumMatch) {
      const fid = forumMatch[1];
      console.log(`📍 从forum路径识别版块: fid=${fid}`);
      return fid;
    }

    // 方式3: 从thread URL中推断(访问页面时检查页面元素)
    try {
      const breadcrumbs = document.querySelectorAll('.z a[href*="forum"]');
      const forumIds = [];

      for (const breadcrumb of breadcrumbs) {
        // 尝试从 forum-数字- 格式提取
        const breadcrumbMatch1 = breadcrumb.href.match(/forum-(\d+)-/);
        if (breadcrumbMatch1) {
          const fid = breadcrumbMatch1[1];
          const name = breadcrumb.textContent.trim();
          forumIds.push({ fid, name, element: breadcrumb });
          continue;
        }
        // 尝试从 fid= 参数提取
        const breadcrumbMatch2 = breadcrumb.href.match(/fid=(\d+)/);
        if (breadcrumbMatch2) {
          const fid = breadcrumbMatch2[1];
          const name = breadcrumb.textContent.trim();
          forumIds.push({ fid, name, element: breadcrumb });
        }
      }

      // ✨ 选择最后一个版块(最精确的子版块)
      if (forumIds.length > 0) {
        const lastForum = forumIds[forumIds.length - 1];
        console.log(
          `📍 从面包屑导航识别版块: fid=${lastForum.fid} (${lastForum.name})`
        );
        if (forumIds.length > 1) {
          console.log(
            `   💡 跳过了 ${forumIds.length - 1} 个父级版块,选择最精确的子版块`
          );
        }
        return lastForum.fid;
      }
    } catch (e) {
      console.log("⚠️ 无法从面包屑导航识别版块:", e.message);
    }

    // 方式4: 从当前页面的返回链接中提取
    try {
      const backLinks = document.querySelectorAll('a[href*="forum"]');
      for (const backLink of backLinks) {
        const linkMatch = backLink.href.match(/fid=(\d+)/);
        if (linkMatch) {
          const fid = linkMatch[1];
          console.log(`📍 从返回链接识别版块: fid=${fid}`);
          return fid;
        }
      }
    } catch (e) {
      // ignore
    }

    // 默认返回中文百合漫画区 (fid=30)
    console.log("📍 无法识别版块,默认使用中文百合漫画区 (fid=30)");
    return "30";
  }

  // 获取版块名称(用于日志显示)
  function getForumName(fid) {
    const forumNames = {
      30: "中文百合漫画区",
      37: "百合漫画图源区",
    };
    return forumNames[fid] || `版块${fid}`;
  }

  // 与本贴相似度匹配检查
  function containsSimilarWords(title1, title2) {
    const words1 = title1
      .toLowerCase()
      .split(/[\s\-_]+/)
      .filter((w) => w.length > 1);
    const words2 = title2
      .toLowerCase()
      .split(/[\s\-_]+/)
      .filter((w) => w.length > 1);

    console.log(`🔍 词汇比较: "${title1}" vs "${title2}"`);
    console.log(`   词汇1: [${words1.join(", ")}]`);
    console.log(`   词汇2: [${words2.join(", ")}]`);

    let commonWords = 0;
    const matchedPairs = [];

    for (const word1 of words1) {
      for (const word2 of words2) {
        if (word1.includes(word2) || word2.includes(word1)) {
          commonWords++;
          matchedPairs.push(`"${word1}" ⟷ "${word2}"`);
          break;
        }
      }
    }

    const threshold = Math.min(2, Math.min(words1.length, words2.length));
    const isMatch = commonWords >= threshold;

    console.log(`   匹配词汇: ${matchedPairs.join(", ")}`);
    console.log(
      `   匹配数量: ${commonWords}/${threshold} (需要: ${threshold})`
    );
    console.log(`   结果: ${isMatch ? "✅ 匹配" : "❌ 不匹配"}`);

    return isMatch;
  }

  /*** 悬浮按钮 ***/
  const ioniconsModule = document.createElement("script");
  ioniconsModule.type = "module";
  ioniconsModule.src =
    "https://unpkg.com/[email protected]/dist/ionicons/ionicons.esm.js";
  document.head.appendChild(ioniconsModule);

  const ioniconsNomodule = document.createElement("script");
  ioniconsNomodule.noModule = true;
  ioniconsNomodule.src =
    "https://unpkg.com/[email protected]/dist/ionicons/ionicons.js";
  document.head.appendChild(ioniconsNomodule);

  const fontLink = document.createElement("link");
  fontLink.href =
    "https://fonts.googleapis.com/css2?family=Poppins&display=swap";
  fontLink.rel = "stylesheet";
  document.head.appendChild(fontLink);

  const button = document.createElement("button");
  button.id = "reader-toggle";
  button.innerHTML = `
    <span class="icon"><ion-icon name="book-outline"></ion-icon></span>
    <span class="title">进入阅读模式</span>
  `;

  document.body.appendChild(button);

  GM_addStyle(BUTTON_CSS);

  // 添加快捷键:Ctrl+Shift+R 切换阅读模式
  document.addEventListener("keydown", (e) => {
    if (e.ctrlKey && e.shiftKey && e.key.toLowerCase() === "r") {
      e.preventDefault();
      if (!isReading) enterReader();
      else exitReader();
    }
  });

  button.addEventListener("click", () => {
    if (!isReading) {
      enterReader();
      setAutoReaderStatus(true); // 开启全局阅读器模式
    } else {
      exitReader();
    }
  });

  /*** 目录功能 ***/
  async function showDirectoryModal() {
    // 创建遮罩层
    const overlay = document.createElement("div");
    overlay.id = "directory-overlay";

    // 创建侧边栏
    const sidebar = document.createElement("div");
    sidebar.id = "directory-sidebar";
    sidebar.innerHTML = `
      <div id="directory-header">
        <div id="directory-title">目录</div>
        <button id="directory-close">×</button>
      </div>
      <div id="directory-content">
        <div id="directory-loading">正在搜索相关章节...</div>
        <ul id="directory-list" style="display: none;"></ul>
        <div id="directory-empty" style="display: none;">未找到相关章节</div>
      </div>
    `;

    document.body.appendChild(overlay);
    document.body.appendChild(sidebar);

    // 显示遮罩和侧边栏
    setTimeout(() => {
      overlay.classList.add("show");
      sidebar.classList.add("open");
    }, 10);

    // 关闭按钮事件
    const closeBtn = sidebar.querySelector("#directory-close");
    closeBtn.addEventListener("click", closeSidebar);

    // 点击遮罩关闭
    overlay.addEventListener("click", closeSidebar);

    function closeSidebar() {
      overlay.classList.remove("show");
      sidebar.classList.remove("open");

      setTimeout(() => {
        overlay.remove();
        sidebar.remove();
      }, 300);
    }

    // 开始搜索目录
    loadSeriesDirectory(sidebar);
  }

  async function loadSeriesDirectory(modal) {
    console.log("📖 开始加载系列目录...");

    // 如果已有预加载的目录,直接使用
    if (seriesDirectory.length > 0) {
      console.log("✅ 使用预加载的目录:", seriesDirectory.length, "个章节");
      displayDirectory(modal, seriesDirectory);
      return;
    }

    // 如果预加载目录为空,先检查持久化缓存
    if (originalSeriesTitle || savedThreadTitle) {
      const titleToCheck = originalSeriesTitle || savedThreadTitle;
      const seriesKey = generateSeriesKey(titleToCheck);
      const cachedDirectory = getCachedDirectory(seriesKey);
      if (cachedDirectory && cachedDirectory.length > 0) {
        console.log(
          "✅ 从持久化缓存加载目录:",
          cachedDirectory.length,
          "个章节"
        );
        seriesDirectory = cachedDirectory;
        displayDirectory(modal, seriesDirectory);
        return;
      }
    }

    // 尝试多种方式获取页面标题
    let titleElement = null;
    let threadTitle = "";

    console.log("🔍 尝试查找页面标题...");

    // 简化标题提取逻辑
    const h1Element = document.querySelector("h1.ts");
    if (h1Element) {
      const subjectSpan = h1Element.querySelector(
        '#thread_subject, span[id^="thread_subject"]'
      );
      if (subjectSpan) {
        threadTitle = subjectSpan.textContent.trim();
        console.log(`✅ 找到标题: "${threadTitle}"`);
      } else {
        threadTitle = h1Element.textContent.trim();
        console.log(`✅ 找到h1文本: "${threadTitle}"`);
      }
    } else {
      // 兜底:使用原有逻辑
      const titleSelectors = [
        "#thread_subject",
        'span[id^="thread_subject"]',
        ".ts a",
        "h1.ts",
        ".ntn a",
        ".ntn",
        ".bm .mbm h1",
        ".ts",
      ];

      for (const selector of titleSelectors) {
        titleElement = document.querySelector(selector);
        if (titleElement) {
          threadTitle = titleElement.textContent.trim();
          console.log(`✅ 找到标题元素: "${selector}" -> "${threadTitle}"`);
          break;
        } else {
          console.log(`❌ 未找到: "${selector}"`);
        }
      }
    }

    // 在阅读模式下,优先使用原始系列标题进行目录搜索(避免使用错误的标题)
    if (isReading) {
      if (originalSeriesTitle) {
        threadTitle = originalSeriesTitle;
        console.log("🎯 阅读模式下使用原始系列标题:", threadTitle);
      } else if (savedThreadTitle) {
        threadTitle = savedThreadTitle;
        console.log("📦 阅读模式下使用预存标题:", threadTitle);
      }
    } else if (!threadTitle && originalSeriesTitle) {
      threadTitle = originalSeriesTitle;
      console.log("🎯 使用原始系列标题:", threadTitle);
    } else if (!threadTitle && savedThreadTitle) {
      threadTitle = savedThreadTitle;
      console.log("📦 使用预存标题:", threadTitle);
    }
    // 如果还是没找到,尝试从页面title获取
    if (!threadTitle) {
      const pageTitle = document.title;
      console.log("🔍 尝试从页面title获取:", pageTitle);

      // 移除论坛名称等后缀,只保留漫画标题部分
      if (pageTitle.includes(" - ")) {
        threadTitle = pageTitle.split(" - ")[0].trim();
        console.log("📝 从页面title提取:", threadTitle);
      } else {
        threadTitle = pageTitle.replace(/\s*-\s*百合会.*$/, "").trim();
        console.log("📝 清理后的标题:", threadTitle);
      }
    }

    const currentThreadId = getCurrentThreadId();

    console.log("📋 当前页面信息:");
    console.log("   标题元素:", titleElement ? "✅ 找到" : "❌ 未找到");
    console.log("   页面标题:", threadTitle);
    console.log("   线程ID:", currentThreadId);

    // 调试:打印页面中所有可能的标题元素
    if (!titleElement) {
      console.log("🔍 调试:查找所有可能的标题元素...");

      const allH1 = document.querySelectorAll("h1");
      const allSpans = document.querySelectorAll(
        'span[id*="thread"], span[id*="subject"]'
      );
      const allTs = document.querySelectorAll(".ts, .ntn");

      console.log("  - h1元素数量:", allH1.length);
      allH1.forEach((h1, index) => {
        if (index < 5) {
          console.log(
            `    h1[${index}]: "${h1.textContent
              .trim()
              .substring(0, 50)}" (class: ${h1.className})`
          );
        }
      });

      console.log("  - 包含thread/subject的span数量:", allSpans.length);
      allSpans.forEach((span, index) => {
        if (index < 5) {
          console.log(
            `    span[${index}]: id="${span.id}", text="${span.textContent
              .trim()
              .substring(0, 50)}"`
          );
        }
      });

      console.log("  - .ts/.ntn元素数量:", allTs.length);
      allTs.forEach((ts, index) => {
        if (index < 5) {
          console.log(
            `    ts[${index}]: class="${ts.className}", text="${ts.textContent
              .trim()
              .substring(0, 50)}"`
          );
        }
      });
    }

    if (!threadTitle) {
      console.log("❌ 页面标题为空,显示空状态");
      showDirectoryEmpty(modal);
      return;
    }

    try {
      console.log("🔍 开始搜索目录...");
      const directory = await searchSeriesDirectory(threadTitle);

      console.log("📊 搜索结果:", directory.length, "个章节");

      if (directory.length === 0) {
        console.log("❌ 未找到章节,显示空状态");
        showDirectoryEmpty(modal);
        return;
      }

      console.log("✅ 找到章节,开始显示目录界面");
      displayDirectory(modal, directory, threadTitle);
    } catch (error) {
      console.error("❌ 加载目录失败:", error);
      console.error("错误详情:", error.message);
      console.error("错误堆栈:", error.stack);
      showDirectoryEmpty(modal, "加载失败,请稍后重试");
    }
  }

  function displayDirectory(modal, directory, threadTitle = "") {
    // 显示目录
    const loading = modal.querySelector("#directory-loading");
    const list = modal.querySelector("#directory-list");
    const title = modal.querySelector("#directory-title");

    loading.style.display = "none";
    list.style.display = "block";

    // 获取页面标题(如果没有提供)
    if (!threadTitle) {
      if (savedThreadTitle) {
        threadTitle = savedThreadTitle;
      } else {
        threadTitle = document.title.split(" - ")[0] || "漫画目录";
      }
    }

    // 简化标题,只显示"目录"
    title.textContent = "目录";

    console.log("📝 目录标题: 目录");

    list.innerHTML = "";

    const currentThreadId = getCurrentThreadId();

    directory.forEach((item, index) => {
      const li = document.createElement("li");
      const link = document.createElement("a");
      link.className = "directory-item";
      if (item.source === "mainpost") {
        link.classList.add("mainpost");
      }

      link.href = item.url;

      // 为主楼提取的目录优化显示格式
      if (item.source === "mainpost") {
        let displayText = item.title;

        // 识别不同的章节格式并优化显示
        if (item.title.match(/^\d{1,2}$/)) {
          displayText = `第${String(item.title).padStart(2, "0")}话`;
        } else if (item.title.match(/^\d+\.\d+$/)) {
          displayText = `第${item.title}话`;
        } else if (item.title.match(/^\d+[话話章节節回卷篇]/i)) {
          displayText = item.title;
        } else if (item.title.match(/卷|番外|彩页|特典/i)) {
          displayText = item.title;
        } else if (item.title.match(/^0\d+$/)) {
          displayText = `第${item.title}话`;
        }

        link.textContent = displayText;
      } else {
        link.textContent = item.title;
      }

      // 添加点击事件处理,无缝加载新章节内容
      link.addEventListener("click", async (e) => {
        e.preventDefault();
        console.log("🔄 正在加载章节:", item.title);

        // 关闭侧边栏
        const overlay = document.getElementById("directory-overlay");
        const sidebar = document.getElementById("directory-sidebar");
        if (overlay && sidebar) {
          overlay.classList.remove("show");
          sidebar.classList.remove("open");
          setTimeout(() => {
            overlay.remove();
            sidebar.remove();
          }, 300);
        }

        // 无缝加载新章节内容
        await loadNewChapter(item.url, item.title);
      });

      // 高亮当前章节
      if (item.threadId === currentThreadId) {
        link.classList.add("current");
        console.log(`⭐ 当前章节: ${item.title}`);
      }

      li.appendChild(link);
      list.appendChild(li);

      console.log(`📄 章节 ${index + 1}: ${item.title}`);
    });

    console.log("✅ 目录界面显示完成");
  }

  function showDirectoryEmpty(modal, message = "未找到相关章节") {
    const loading = modal.querySelector("#directory-loading");
    const empty = modal.querySelector("#directory-empty");

    loading.style.display = "none";
    empty.style.display = "block";
    empty.textContent = message;
  }

  function getCurrentThreadId() {
    const href = window.location.href;
    let match = href.match(/thread-(\d+)-/);
    if (match) {
      return match[1];
    }

    try {
      const url = new URL(href);
      const tidParam = url.searchParams.get("tid");
      if (tidParam) {
        return tidParam;
      }
    } catch (e) {
      // ignore
    }

    match = href.match(/[?&]tid=(\d+)/);
    return match ? match[1] : null;
  }

  // 清理所有缓存,并清空内存缓存
  function clearAllCache() {
    console.log("🧹 清理所有目录缓存 (session + GM) ...");
    let cleanedCount = 0;

    // 1) 清理 sessionStorage 前缀条目
    try {
      const sKeys = Object.keys(sessionStorage).filter((k) =>
        k.startsWith(CACHE_PREFIX)
      );
      sKeys.forEach((k) => {
        try {
          sessionStorage.removeItem(k);
          cleanedCount++;
        } catch (e) {}
      });
    } catch (e) {}

    // 2) 清理 GM_* 条目(通过索引枚举)
    try {
      if (
        typeof GM_getValue === "function" &&
        typeof GM_setValue === "function"
      ) {
        let idx = GM_getValue(GM_INDEX_KEY, undefined);
        if (typeof idx === "string") {
          try {
            idx = JSON.parse(idx);
          } catch (e) {
            idx = [];
          }
        }
        if (!Array.isArray(idx)) idx = [];

        for (const key of idx) {
          try {
            storageRemove(key);
            cleanedCount++;
          } catch (e) {}
        }

        try {
          GM_setValue(GM_INDEX_KEY, []);
        } catch (e) {}
      }
    } catch (e) {}

    // 3) 清理内存缓存
    try {
      if (searchCache && searchCache.clear) searchCache.clear();
    } catch (e) {}
    try {
      if (directoryMemoryCache && directoryMemoryCache.clear)
        directoryMemoryCache.clear();
    } catch (e) {}

    console.log(`✅ 已清理 ${cleanedCount} 个持久化缓存条目,且已清空内存缓存`);
    return cleanedCount;
  }

  // 查看缓存状态(session + GM 索引 + 内存统计)
  function viewCacheStatus() {
    console.log("📊 缓存状态统计:");

    // sessionStorage 条目
    let sessionList = [];
    try {
      sessionList = Object.keys(sessionStorage).filter((k) =>
        k.startsWith(CACHE_PREFIX)
      );
    } catch (e) {
      sessionList = [];
    }

    // GM 索引条目
    let gmList = [];
    try {
      if (typeof GM_getValue === "function") {
        let idx = GM_getValue(GM_INDEX_KEY, undefined);
        if (typeof idx === "string") {
          try {
            idx = JSON.parse(idx);
          } catch (e) {
            idx = [];
          }
        }
        if (Array.isArray(idx)) gmList = idx.slice();
      }
    } catch (e) {
      gmList = [];
    }

    console.log(`📦 sessionStorage 缓存数量: ${sessionList.length}`);
    console.log(`📦 GM_* 缓存数量 (index): ${gmList.length}`);
    console.log(
      `🧠 内存缓存 (searchCache): ${
        typeof searchCache !== "undefined" && searchCache.size !== undefined
          ? searchCache.size
          : "未知"
      }`
    );
    console.log(
      `🧠 内存目录缓存 (directoryMemoryCache): ${
        typeof directoryMemoryCache !== "undefined" &&
        directoryMemoryCache.size !== undefined
          ? directoryMemoryCache.size
          : "未知"
      }`
    );

    if (sessionList.length > 0) {
      console.log("\n📋 sessionStorage 详情:");
      sessionList.forEach((k, i) => {
        try {
          const obj = JSON.parse(sessionStorage.getItem(k));
          const age = Math.round(
            (Date.now() - (obj.ts || obj.timestamp || 0)) / (1000 * 60 * 60)
          );
          const count = obj.d || obj.data ? (obj.d || obj.data).length : "未知";
          console.log(
            `   ${i + 1}. ${k.replace(
              CACHE_PREFIX + "session-",
              ""
            )}: ${count}章, ${age}小时前`
          );
        } catch (e) {
          console.log(`   ${i + 1}. ${k} (无法解析)`);
        }
      });
    }

    if (gmList.length > 0) {
      console.log("\n📋 GM_* 详情 (由索引列出):");
      gmList.forEach((k, i) => {
        try {
          const obj = storageGet(k, null);
          const age = obj
            ? Math.round(
                (Date.now() - (obj.ts || obj.timestamp || 0)) / (1000 * 60 * 60)
              )
            : "未知";
          const count = obj
            ? obj.d || obj.data
              ? (obj.d || obj.data).length
              : "未知"
            : "未知";
          console.log(
            `   ${i + 1}. ${k.replace(
              CACHE_PREFIX,
              ""
            )}: ${count}章, ${age}小时前`
          );
        } catch (e) {
          console.log(`   ${i + 1}. ${k} (无法读取)`);
        }
      });
    }

    return {
      sessionCount: sessionList.length,
      gmCount: gmList.length,
      memorySearchCount:
        typeof searchCache !== "undefined" && searchCache.size !== undefined
          ? searchCache.size
          : null,
      memoryDirCount:
        typeof directoryMemoryCache !== "undefined" &&
        directoryMemoryCache.size !== undefined
          ? directoryMemoryCache.size
          : null,
    };
  }

  // 智能图片收集函数:强力过滤 + 智能兜底
  function collectImagesFromDocument(doc = document) {
    try {
      // URL 标准化
      const normalizeUrl = (u) => {
        if (!u) return null;
        u = String(u).trim();
        if (!u) return null;
        if (u.startsWith("//")) u = (location.protocol || "https:") + u;
        if (!/^https?:\/\//i.test(u))
          u = "https://bbs.yamibo.com/" + u.replace(/^\/+/, "");
        u = u.replace(/^https?:\/\/https?:\/\//, "https://");
        return u;
      };

      // 强力噪音判定
      const isIgnoredImage = (img, src) => {
        try {
          if (!src && !img) return true;
          if (BLOCK_IMG_REGEX.test(src)) return true;

          // === 1. 文件名与路径黑名单 (精准打击) ===
          const blockKeywords = [
            // 基础UI
            "/uc_server/data/avatar/",
            "avatar",
            "user_avatar", // 头像
            "static/image/common/",
            "static/image/smiley/", // 系统图标/表情
            "template/", // 模板图片 (如 userinfo.gif, forumlink.gif)

            // 具体文件名特征
            "none.gif",
            "loading.gif",
            "logo.png",
            "logo.gif",
            "qq.gif",
            "qq_big.gif",
            "qq_group", // QQ相关
            "userinfo.gif",
            "forumlink.gif", // 用户资料/网站链接
            "online_admin",
            "online_member",
            "online_team", // 在线状态
            "icon_quote",
            "collapse",
            "expand", // 引用/折叠图标

            // 评分与功能
            "rating",
            "score",
            "grade",
            "star",
            "magic",
          ];

          if (blockKeywords.some((kw) => s.includes(kw))) return true;

          // === 2. CSS 类名/ID 过滤 ===
          const cls = img && img.className ? img.className : "";
          if (/avatar|logo|vm|authicn/i.test(cls)) return true;
          const id = img && img.id ? img.id : "";
          if (/authicon|logo/i.test(id)) return true;

          // === 3. 容器过滤 (防止误杀正文,但要排除侧边栏等) ===
          if (
            img &&
            (img.closest(".postrate") ||
              img.closest(".post-ratings") ||
              img.closest(".postratedby") ||
              img.closest(".poster") ||
              img.closest(".author") ||
              img.closest(".avatar") ||
              img.closest(".user") ||
              img.closest(".pls") || // .pls 是用户信息侧边栏
              img.closest(".p_pop") || // 弹出菜单
              img.closest("#ft") || // 页脚
              img.closest(".po") || // 帖子底部操作栏
              img.closest(".a_pr")) // 广告位
          ) {
            return true;
          }

          // === 4. 尺寸过滤 (针对未被上述规则命中的漏网之鱼) ===
          const wAttr = img && (img.getAttribute("width") || img.width);
          const hAttr = img && (img.getAttribute("height") || img.height);
          const w = wAttr ? parseInt(wAttr, 10) : 0;
          const h = hAttr ? parseInt(hAttr, 10) : 0;

          // 只有当宽高都非常明确且很小时才过滤 (小于 100px 通常是图标)
          // 漫画图片通常宽度远大于 100
          if (w > 0 && h > 0 && (w < 100 || h < 100)) return true;

          // 还可以检测 style 属性中的宽高
          if (img && img.style) {
            const styleW = parseInt(img.style.width || "0");
            const styleH = parseInt(img.style.height || "0");
            if (styleW > 0 && styleH > 0 && (styleW < 100 || styleH < 100))
              return true;
          }

          return false;
        } catch (e) {
          return false;
        }
      };

      // 优先选择器 (大图通常有的属性)
      const preferredSelectors = [
        "ignore_js_op img",
        ".savephotop img",
        "div.t_f img",
        "td.t_f img",
        ".pcb img",
        "img[zoomfile]",
        "img[file]",
        "img[aid]",
        "img.zoom",
      ];

      // 辅助:执行收集
      const doCollect = (context, selectors) => {
        const list = [];
        const visited = new Set();

        // 尝试优先选择器
        for (const sel of selectors) {
          const imgs = context.querySelectorAll(sel);
          imgs.forEach((img) => {
            const u = normalizeUrl(
              img.getAttribute("zoomfile") ||
                img.getAttribute("file") ||
                img.getAttribute("src")
            );
            if (u && !visited.has(u) && !isIgnoredImage(img, u)) {
              visited.add(u);
              list.push(u);
            }
          });
        }

        // 如果优先选择器没找到,尝试所有图片但严格过滤
        if (list.length === 0) {
          const allImgs = context.querySelectorAll("img");
          allImgs.forEach((img) => {
            const u = normalizeUrl(img.getAttribute("src"));
            // 这里只收录有 zoomfile 或看起来很大的图
            const hasZoom =
              img.getAttribute("zoomfile") || img.getAttribute("file");
            if (u && !visited.has(u) && !isIgnoredImage(img, u)) {
              if (hasZoom) {
                visited.add(u);
                list.push(u);
              }
            }
          });
        }
        return list;
      };

      let resultsA = [];
      let resultsB = [];

      // === 策略 A: 按楼主 UID 过滤 (增强版) ===
      let authorUid = null;

      // 1. 查找所有帖子楼层
      const allPostDivs = Array.from(doc.querySelectorAll('div[id^="post_"]'));
      // 过滤干扰项,只留真正楼层
      const realPosts = allPostDivs.filter(
        (p) => p.id && /^post_\d+$/.test(p.id)
      );

      // 2. 尝试从主楼 (一楼) 提取 UID
      if (realPosts.length > 0) {
        const firstPost = realPosts[0];
        const links = firstPost.querySelectorAll(
          'a[href*="uid"], a[href*="space"]'
        );
        for (const a of links) {
          const href = a.getAttribute("href");
          const m = href.match(/(?:uid[=-]|space-uid-)(\d+)/);
          if (m) {
            authorUid = m[1];
            break;
          }
        }
      }

      // 3. 备用: 全局查找 "楼主" 标识
      if (!authorUid) {
        const louzhu = doc.querySelector('.authicn[title="楼主"], .louzhu');
        if (louzhu) {
          const parent = louzhu.closest(".pi") || louzhu.closest(".authi");
          if (parent) {
            const a = parent.querySelector('a[href*="uid"]');
            if (a) {
              const m = a.href.match(/(?:uid[=-]|space-uid-)(\d+)/);
              if (m) authorUid = m[1];
            }
          }
        }
      }

      // 4. 如果找到了 UID,收集所有该 UID 的楼层
      if (authorUid) {
        const authorPosts = realPosts.filter((p) => {
          // 检查楼层内的用户信息链接是否包含该 UID
          // 这种检查方式比 innerHTML includes 更快且更准
          const userLink = p.querySelector(
            `a[href*="uid=${authorUid}"], a[href*="uid-${authorUid}"]`
          );
          return !!userLink;
        });

        authorPosts.forEach((p) => {
          const imgs = doCollect(p, preferredSelectors);
          resultsA.push(...imgs);
        });
      }

      // === 策略 B: 兜底模式 (扫描所有内容区域 .t_f) ===
      // Discuz 的帖子内容通常都在 .t_f 或 .pcb 中
      const contentAreas = doc.querySelectorAll(".t_f, .pcb");
      contentAreas.forEach((area) => {
        const imgs = doCollect(area, preferredSelectors);
        resultsB.push(...imgs);
      });

      // === 决策 ===

      // 去重合并
      const uniqueA = [...new Set(resultsA)];
      const uniqueB = [...new Set(resultsB)];

      console.log(
        `📊 图片提取: 楼主模式=${uniqueA.length}, 兜底模式=${uniqueB.length}`
      );

      if (uniqueA.length > 0) {
        // 如果楼主模式找到了图,通常比较准
        return uniqueA;
      } else {
        // 否则使用兜底模式
        if (uniqueB.length > 0) {
          console.log("⚠️ 楼主模式未找到图片,使用兜底结果");
        }
        return uniqueB;
      }
    } catch (e) {
      console.warn("collectImagesFromDocument error", e);
      return [];
    }
  }
  // ====== localForage 图片缓存工具函数 ======
  async function getCachedImageUrl(imgUrl) {
    const cached = await localforage.getItem(imgUrl);
    if (cached) {
      return URL.createObjectURL(cached);
    }
    const resp = await fetch(imgUrl);
    const blob = await resp.blob();
    await localforage.setItem(imgUrl, blob);
    return URL.createObjectURL(blob);
  }
  // 初始化时检查全局阅读器状态
  function initAutoReader() {
    autoReaderEnabled = checkAutoReaderStatus();
    console.log("🚀 初始化全局阅读器状态:", autoReaderEnabled);

    // 清理过期缓存
    cleanExpiredCache();

    // 检查是否有漫画内容,优先检测主楼
    const comicImages = collectImagesFromDocument(document);
    if (comicImages && comicImages.length > 0) {
      console.log("🎯 检测到漫画内容,立即开始预取目录...");

      // 立即开始预取目录信息,不管是否进入阅读模式
      preloadDirectoryInfo();

      // 如果全局阅读器开启,自动进入阅读模式
      if (autoReaderEnabled && !isReading) {
        console.log("🎯 自动进入阅读模式");
        setTimeout(() => {
          if (button && button.parentNode) {
            button.remove();
          }
          enterReader();
        }, 1000);
      }
    } else {
      console.log("🔍 尝试预取目录信息...");
      preloadDirectoryInfo();
    }
  }
  // 预加载目录信息
  async function preloadDirectoryInfo() {
    if (seriesDirectory.length > 0) {
      console.log("� 目录已预加载,跳过重复加载");
      return;
    }

    try {
      console.log("�📚 开始预加载目录信息...");

      // 获取当前页面标题
      let threadTitle = "";
      const h1Element = document.querySelector("h1.ts");
      if (h1Element) {
        const subjectSpan = h1Element.querySelector(
          '#thread_subject, span[id^="thread_subject"]'
        );
        if (subjectSpan) {
          threadTitle = subjectSpan.textContent.trim();
        } else {
          threadTitle = h1Element.textContent.trim();
        }
      } else {
        const titleElement = document.querySelector(
          '#thread_subject, span[id^="thread_subject"]'
        );
        if (titleElement) {
          threadTitle = titleElement.textContent.trim();
        } else {
          const pageTitle = document.title;
          threadTitle = pageTitle.includes(" - ")
            ? pageTitle.split(" - ")[0].trim()
            : pageTitle.replace(/\s*-\s*百合会.*$/, "").trim();
        }
      }

      if (threadTitle) {
        // 保存标题信息
        savedThreadTitle = threadTitle;
        originalSeriesTitle = threadTitle;

        console.log("📖 开始预取目录,使用标题:", threadTitle);

        // 先检查持久化缓存
        const seriesKey = generateSeriesKey(threadTitle);
        const cachedDirectory = getCachedDirectory(seriesKey);
        if (cachedDirectory && cachedDirectory.length > 0) {
          console.log(
            "✅ 从持久化缓存预加载完成:",
            cachedDirectory.length,
            "个章节"
          );
          seriesDirectory = cachedDirectory;
          return;
        }

        // 如果没有缓存,立即开始异步搜索目录
        searchSeriesDirectory(threadTitle)
          .then((directory) => {
            if (directory && directory.length > 0) {
              seriesDirectory = directory;
              console.log("✅ 预加载完成,获得", directory.length, "个章节");
            } else {
              console.log("⚠️ 预加载未找到目录");
            }
          })
          .catch((error) => {
            console.warn("⚠️ 预加载目录失败:", error.message);
          });
        console.log("🚀 预加载任务已启动,后台进行中...");
      } else {
        console.log("⚠️ 无法获取页面标题,跳过预加载");
      }
    } catch (error) {
      console.warn("⚠️ 预加载目录失败:", error.message);
    }
  }

  // 页面加载完成后初始化
  if (document.readyState === "loading") {
    document.addEventListener("DOMContentLoaded", () => {
      setTimeout(initAutoReader, 1000); // 确保页面元素完全加载
    });
  } else {
    setTimeout(initAutoReader, 1000); // 给页面充足时间渲染
  }

  /*** 进入阅读模式 ***/
  function enterReader() {
    console.log("[enterReader] called");
    isReading = true;

    // 保存原始页面HTML和标题
    readerStartUrl = window.location.href;
    originalPageHTML = document.body.innerHTML;
    originalPageTitle = document.title;
    console.log("💾 已保存原始页面内容");

    button.remove();

    // 修复 Discuz 原生 JS (showWindow) 依赖的容器
    if (!document.getElementById("append_parent")) {
      const ap = document.createElement("div");
      ap.id = "append_parent";
      document.body.appendChild(ap);
    }
    if (!document.getElementById("ajaxwaitid")) {
      const aw = document.createElement("div");
      aw.id = "ajaxwaitid";
      aw.style.display = "none";
      document.body.appendChild(aw);
    }

    // 在替换页面内容之前,先提取并保存标题
    console.log("🔍 进入阅读模式,保存页面标题...");

    // 简化标题提取逻辑
    const h1Element = document.querySelector("h1.ts");
    if (h1Element) {
      const subjectSpan = h1Element.querySelector(
        '#thread_subject, span[id^="thread_subject"]'
      );
      if (subjectSpan) {
        savedThreadTitle = subjectSpan.textContent.trim();
        console.log("💾 保存标题:", savedThreadTitle);
      } else {
        savedThreadTitle = h1Element.textContent.trim();
        console.log("💾 保存h1文本:", savedThreadTitle);
      }
    } else {
      // 兜底:尝试其他方式
      const threadTitleElement = document.querySelector(
        '#thread_subject, span[id^="thread_subject"]'
      );
      if (threadTitleElement) {
        savedThreadTitle = threadTitleElement.textContent.trim();
        console.log("💾 保存标题元素:", savedThreadTitle);
      } else {
        // 最后兜底用<title>
        const pageTitle = document.title;
        savedThreadTitle = pageTitle.includes(" - ")
          ? pageTitle.split(" - ")[0].trim()
          : pageTitle.replace(/\s*-\s*百合会.*$/, "").trim();
        console.log("💾 从页面title保存:", savedThreadTitle);
      }
    }

    // 保存原始系列标题,用于后续目录搜索
    originalSeriesTitle = savedThreadTitle;
    console.log("🎯 保存原始系列标题:", originalSeriesTitle);

    // === 图片提取逻辑 ===
    // 只用 collectImagesFromDocument,避免复用噪音图片
    images = collectImagesFromDocument(document)
      .map((u) => {
        let url = String(u);
        if (url.startsWith("//")) url = "https:" + url;
        if (!/^https?:\/\//i.test(url))
          url = "https://bbs.yamibo.com/" + url.replace(/^\/+/, "");
        if (url.startsWith("http://"))
          url = url.replace(/^http:\/\//, "https://");
        url = url.replace(/^https?:\/\/https?:\/\//, "https://");
        return url;
      })
      .filter(Boolean);
    console.log("📸 提取到", images.length, "张图片,URL已处理为HTTPS");
    // 新增:预加载首图,加速 LCP
    if (images.length > 0) {
      const preload = document.createElement("link");
      preload.rel = "preload";
      preload.as = "image";
      preload.href = images[0];
      document.head.appendChild(preload);
    }
    // 检查是否已有预加载的目录数据
    if (seriesDirectory.length > 0) {
      console.log("✅ 使用预加载的目录数据:", seriesDirectory.length, "个章节");
      // 新增:如果在阅读模式,立即启用按钮
      if (isReading) {
        const prevBtn = document.getElementById("cw-prev");
        const nextBtn = document.getElementById("cw-next");
        if (prevBtn) prevBtn.disabled = !seriesDirectory.length;
        if (nextBtn) nextBtn.disabled = !seriesDirectory.length;
      }
    } else {
      // 检查持久化缓存
      const seriesKey = generateSeriesKey(savedThreadTitle);
      const cachedDirectory = getCachedDirectory(seriesKey);
      if (cachedDirectory && cachedDirectory.length > 0) {
        console.log(
          "📦 从持久化缓存加载目录:",
          cachedDirectory.length,
          "个章节"
        );
        seriesDirectory = cachedDirectory;
        // 新增:如果在阅读模式,立即启用按钮
        if (isReading) {
          const prevBtn = document.getElementById("cw-prev");
          const nextBtn = document.getElementById("cw-next");
          if (prevBtn) prevBtn.disabled = !seriesDirectory.length;
          if (nextBtn) nextBtn.disabled = !seriesDirectory.length;
        }
      } else {
        // 缓存也没有,异步搜索
        console.log("🔍 开始异步搜索目录...");
        searchSeriesDirectory(savedThreadTitle)
          .then((directory) => {
            seriesDirectory = directory;
            console.log("📋 异步获取到目录:", seriesDirectory.length, "个章节");
            // 新增:如果在阅读模式,提示用户按钮已可用
            if (isReading) {
              showErrorMessage("目录已加载,上一话/下一话可用");
              const prevBtn = document.getElementById("cw-prev");
              const nextBtn = document.getElementById("cw-next");
              if (prevBtn) prevBtn.disabled = !seriesDirectory.length;
              if (nextBtn) nextBtn.disabled = !seriesDirectory.length;
            }
          })
          .catch((error) => {
            console.error("❌ 异步获取目录失败:", error);
          });
      }
    }

    // 渲染阅读器UI
    document.body.innerHTML = `
      <button id="cw-exit" data-tooltip="退出阅读模式">⏻</button>
      <div id="cw-toolbar">
        <button id="cw-full" data-tooltip="全屏模式 (F)">⛶</button>
        <button id="cw-bg" data-tooltip="切换背景色 (B)">◐</button>
        <div id="cw-zoom">
          <button id="cw-zoom-in" data-tooltip="放大图片 (+)">﹢</button>
          <button id="cw-zoom-out" data-tooltip="缩小图片 (-)">﹣</button>
        </div>
        <button id="cw-directory" data-tooltip="查看目录 (M)">☰</button>
        <button id="cw-prev" data-tooltip="上一话">≪</button>
        <button id="cw-next" data-tooltip="下一话">≫</button>
        <button id="cw-favorite" data-tooltip="收藏">☆</button>
        <button id="cw-comment" data-tooltip="去评论">❞</button>
      </div>
      <div id="cw-container">
        <div id="cw-title-bar" class="cw-title-bar">${savedThreadTitle}</div>
        <div id="cw-image-list"></div>
      </div>
      <div id="cw-bottom-bar">
        <span id="cw-page-info">1/${images.length}</span>
        <button id="cw-to-top" data-tooltip="返回顶部 (T)">⬆</button>
      </div>
    `;
    // 重新插入阅读器样式和依赖脚本
    if (!document.getElementById("yamibo-reader-style")) {
      const styleElement = GM_addStyle(READER_CSS);
      if (styleElement) styleElement.id = "yamibo-reader-style";
    }
    if (!document.getElementById("yamibo-ionicons-module")) {
      const ioniconsModule = document.createElement("script");
      ioniconsModule.type = "module";
      ioniconsModule.src =
        "https://unpkg.com/[email protected]/dist/ionicons/ionicons.esm.js";
      ioniconsModule.id = "yamibo-ionicons-module";
      document.head.appendChild(ioniconsModule);
    }
    if (!document.getElementById("yamibo-ionicons-nomodule")) {
      const ioniconsNomodule = document.createElement("script");
      ioniconsNomodule.noModule = true;
      ioniconsNomodule.src =
        "https://unpkg.com/[email protected]/dist/ionicons/ionicons.js";
      ioniconsNomodule.id = "yamibo-ionicons-nomodule";
      document.head.appendChild(ioniconsNomodule);
    }
    if (!document.getElementById("yamibo-font-link")) {
      const fontLink = document.createElement("link");
      fontLink.href =
        "https://fonts.googleapis.com/css2?family=Poppins&display=swap";
      fontLink.rel = "stylesheet";
      fontLink.id = "yamibo-font-link";
      document.head.appendChild(fontLink);
    }
    if (!document.getElementById("yamibo-plex-sc-font-link")) {
      const ibmFontLink = document.createElement("link");
      ibmFontLink.rel = "stylesheet";
      ibmFontLink.href =
        "https://cdn.jsdelivr.net/npm/@ibm/plex-sans-sc/css/ibm-plex-sans-sc.min.css";
      ibmFontLink.id = "yamibo-plex-sc-font-link";
      document.head.appendChild(ibmFontLink);
    }
    setTimeout(() => {
      try {
        initReaderUI(); // 渲染图片
        console.log("✅ 阅读器 UI 初始化完成");
      } catch (err) {
        console.error("❌ 初始化阅读器UI失败:", err);
      }
    }, 0);
  }

 // ====== 渲染图片统一函数 ======
  function renderImages(imgList, images) {
    if (!imgList) return;
    imgList.innerHTML = "";
    images.forEach((url, idx) => {
      const img = document.createElement("img");
      img.alt = ""; 
      img.dataset.index = String(idx + 1);
      
      img.onload = function () {
        img.classList.add("loaded"); // ✅ 加载成功后,添加 loaded 类
      };
      
      img.onerror = function () {
        img.style.backgroundImage =
          "url(\"data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' width='100%25' height='100%25'%3E%3Ctext x='50%25' y='50%25' dominant-baseline='middle' text-anchor='middle' font-family='sans-serif' font-size='16' fill='%23ff6b6b'%3E图片加载失败%3C/text%3E%3C/svg%3E\")";
        img.style.backgroundColor = "#f8d7da";
        img.removeAttribute("src");
        img.dataset.error = "1";
      };
      
      imgList.appendChild(img); // 图片添加到 DOM

      getCachedImageUrl(url)
        .then((blobUrl) => {
          img.src = blobUrl;
        })
        .catch(() => {
          img.src = url;
        });
    });
  }

  // ====== 初始化阅读器UI(渲染图片并绑定按钮) ======
  function initReaderUI() {
    console.log("[initReaderUI] start");
    const container = document.getElementById("cw-container");
    if (!container) {
      console.warn("⚠️ cw-container 未找到,无法渲染图片");
      return;
    }
    // 保证有图片列表容器
    let imgList = document.getElementById("cw-image-list");
    if (!imgList) {
      imgList = document.createElement("div");
      imgList.id = "cw-image-list";
      container.appendChild(imgList);
    }
    // 设置初始缩放(使用 CSS 变量控制宽度)
    const applyZoom = () => {
      const vw = Math.max(10, Math.min(90, zoomLevel));
      document.body.style.setProperty("--img-width", `${vw}vw`);
    };
    applyZoom();
    // ===== 新增:插入标题区 =====
    let titleDiv = document.getElementById("cw-title-bar");
    if (!titleDiv) {
      titleDiv = document.createElement("div");
      titleDiv.id = "cw-title-bar";
      titleDiv.className = "cw-title-bar";
      const titleText =
        savedThreadTitle ||
        originalSeriesTitle ||
        document.title.split(" - ")[0] ||
        "漫画标题";
      titleDiv.textContent = titleText;
      container.insertBefore(titleDiv, container.firstChild);
    } else {
      const titleText =
        savedThreadTitle ||
        originalSeriesTitle ||
        document.title.split(" - ")[0] ||
        "漫画标题";
      titleDiv.textContent = titleText;
    }
    // 渲染图片列表
    renderImages(imgList, images);

    // 更新页码显示(靠近视口中心的图片)
    const pageInfo = document.getElementById("cw-page-info");
    function updateCurrentPageInfo() {
      const imgs = container.querySelectorAll("img");
      if (!imgs.length) return;
      if (window.scrollY < 50) {
        if (pageInfo) pageInfo.textContent = `1/${imgs.length}`;
        return;
      }

      const viewportMid = window.innerHeight / 2;
      let current = 1;
      let minDist = Infinity;

      imgs.forEach((img, i) => {
        const rect = img.getBoundingClientRect();
        const imgMidRel = rect.top + rect.height / 2;
        const dist = Math.abs(imgMidRel - viewportMid);

        if (dist < minDist) {
          minDist = dist;
          current = i + 1;
        }
      });

      if (pageInfo) pageInfo.textContent = `${current}/${imgs.length}`;
    }
    // 退出、目录、全屏、背景、缩放、回顶部
    readerEventHandlers.onExitClick = () => exitReader();
    readerEventHandlers.onDirectoryClick = () => showDirectoryModal();
    readerEventHandlers.onFullClick = async () => {
      try {
        if (!document.fullscreenElement)
          await document.documentElement.requestFullscreen();
        else await document.exitFullscreen();
      } catch (e) {
        console.warn("全屏切换失败:", e);
      }
    };
    readerEventHandlers.onBgClick = () => {
      bgIsBlack = !bgIsBlack;
      if (bgIsBlack) {
        document.body.classList.remove("light-bg");
      } else {
        document.body.classList.add("light-bg");
      }
    };
    readerEventHandlers.onZoomIn = () => {
      zoomLevel = Math.min(90, zoomLevel + 5);
      applyZoom();
    };
    readerEventHandlers.onZoomOut = () => {
      zoomLevel = Math.max(10, zoomLevel - 5);
      applyZoom();
    };
    readerEventHandlers.onToTop = () =>
      window.scrollTo({ top: 0, behavior: "smooth" });

    const throttle = (fn, wait = 100) => {
      let t = null;
      return (...args) => {
        if (t) return;
        t = setTimeout(() => {
          fn(...args);
          t = null;
        }, wait);
      };
    };

    readerEventHandlers.updateCurrentPageInfo = throttle(
      updateCurrentPageInfo,
      120
    );

    // 工具栏显示控制
    readerEventHandlers.showTools = function () {
      try {
        document.documentElement.classList.add("tools-visible");
        if (readerToolsTimer) clearTimeout(readerToolsTimer);
        readerToolsTimer = setTimeout(() => {
          document.documentElement.classList.remove("tools-visible");
          readerToolsTimer = null;
        }, 3000);
      } catch (e) {
        // ignore
      }
    };

    readerEventHandlers.onMousemove = readerEventHandlers.showTools;
    readerEventHandlers.onKeydown = readerEventHandlers.showTools;

    readerEventHandlers.onFullscreen = function () {
      if (document.fullscreenElement) {
        readerEventHandlers.showTools();
      } else {
        document.documentElement.classList.remove("tools-visible");
        if (readerToolsTimer) {
          clearTimeout(readerToolsTimer);
          readerToolsTimer = null;
        }
      }
    };

    // ===== 工具栏按钮事件统一绑定 =====
    function initToolbarEvents() {
      const exitBtn = document.getElementById("cw-exit");
      if (exitBtn)
        exitBtn.addEventListener("click", readerEventHandlers.onExitClick);

      const dirBtn = document.getElementById("cw-directory");
      if (dirBtn)
        dirBtn.addEventListener("click", readerEventHandlers.onDirectoryClick);

      const fullBtn = document.getElementById("cw-full");
      if (fullBtn)
        fullBtn.addEventListener("click", readerEventHandlers.onFullClick);

      const bgBtn = document.getElementById("cw-bg");
      if (bgBtn) bgBtn.addEventListener("click", readerEventHandlers.onBgClick);

      const zoomIn = document.getElementById("cw-zoom-in");
      const zoomOut = document.getElementById("cw-zoom-out");
      if (zoomIn)
        zoomIn.addEventListener("click", readerEventHandlers.onZoomIn);
      if (zoomOut)
        zoomOut.addEventListener("click", readerEventHandlers.onZoomOut);

      const toTop = document.getElementById("cw-to-top");
      if (toTop) toTop.addEventListener("click", readerEventHandlers.onToTop);

      // 上一话、下一话、收藏、去评论
      const prevBtn = document.getElementById("cw-prev");
      const nextBtn = document.getElementById("cw-next");
      // 初始时根据目录数据禁用/启用
      if (prevBtn) prevBtn.disabled = !seriesDirectory.length;
      if (nextBtn) nextBtn.disabled = !seriesDirectory.length;
      const favBtn = document.getElementById("cw-favorite");
      const commentBtn = document.getElementById("cw-comment");

      // 上一话
      if (prevBtn)
        prevBtn.onclick = async () => {
          const currentTid = String(getCurrentThreadId());
          const idx = seriesDirectory.findIndex(
            (item) => String(item.threadId) === currentTid
          );
          if (idx > 0) {
            const prevItem = seriesDirectory[idx - 1];
            await loadNewChapter(prevItem.url, prevItem.title);
          } else {
            showErrorMessage("已经是第一话了");
          }
        };

      // 下一话
      if (nextBtn)
        nextBtn.onclick = async () => {
          const currentTid = String(getCurrentThreadId());
          const idx = seriesDirectory.findIndex(
            (item) => String(item.threadId) === currentTid
          );
          if (idx !== -1 && idx < seriesDirectory.length - 1) {
            const nextItem = seriesDirectory[idx + 1];
            await loadNewChapter(nextItem.url, nextItem.title);
          } else {
            showErrorMessage("已经是最后一话了");
          }
        };

      // 收藏
      if (favBtn)
        favBtn.onclick = async () => {
          if (favBtn.dataset.loading) return;
          favBtn.dataset.loading = true;
          favBtn.style.opacity = "0.7";
          // 兼容<i>或纯文本
          const iTag = favBtn.querySelector("i");
          const originalText = iTag ? iTag.innerText : favBtn.innerText;
          if (iTag) iTag.innerText = "...";
          else favBtn.innerText = "...";
          const tid = getCurrentThreadId();
          const href = `/home.php?mod=spacecp&ac=favorite&type=thread&id=${tid}`;
          try {
            const res = await fetch(href + "&infloat=yes&handlekey=k_favorite");
            const text = await res.text();
            if (text.includes("成功") || text.includes("succeed")) {
              showErrorMessage("收藏成功");
              favBtn.classList.add("active");
              if (iTag) iTag.innerText = "★";
              else favBtn.innerText = "★";
            } else if (text.includes("重复") || text.includes("repeat")) {
              showErrorMessage("已收藏");
              if (iTag) iTag.innerText = "★";
              else favBtn.innerText = "★";
            } else {
              showErrorMessage("请求已发送");
              if (iTag) iTag.innerText = originalText;
              else favBtn.innerText = originalText;
            }
          } catch (err) {
            showErrorMessage("请求失败");
            if (iTag) iTag.innerText = originalText;
            else favBtn.innerText = originalText;
          } finally {
            favBtn.dataset.loading = false;
            favBtn.style.opacity = "1";
          }
        };

      // 去评论
      if (commentBtn)
        commentBtn.onclick = () => {
          exitReader();
          setTimeout(() => {
            const fastpost = document.getElementById("fastpostmessage");
            if (fastpost) {
              fastpost.scrollIntoView({ behavior: "smooth", block: "center" });
              fastpost.focus();
            }
          }, 500);
        };
    }

    // ====== 绑定所有工具栏事件 ======
    initToolbarEvents();
    setupRefreshButton();
    // 全局事件绑定
    if (!readerEventsBound) {
      window.addEventListener(
        "scroll",
        readerEventHandlers.updateCurrentPageInfo,
        { passive: true }
      );
      window.addEventListener("mousemove", readerEventHandlers.onMousemove, {
        passive: true,
      });
      window.addEventListener("keydown", readerEventHandlers.onKeydown, {
        passive: true,
      });
      document.addEventListener(
        "fullscreenchange",
        readerEventHandlers.onFullscreen
      );
      readerEventsBound = true;
    }

    // 立即更新一次页码并短暂显示工具栏
    setTimeout(updateCurrentPageInfo, 200);
    readerEventHandlers.showTools();

    // 提供解绑函数,供 exitReader 调用
    unbindReaderEvents = function () {
      try {
        if (!readerEventsBound) return;

        // 移除 DOM 绑定
        try {
          if (exitBtn)
            exitBtn.removeEventListener(
              "click",
              readerEventHandlers.onExitClick
            );
        } catch (e) {}
        try {
          if (dirBtn)
            dirBtn.removeEventListener(
              "click",
              readerEventHandlers.onDirectoryClick
            );
        } catch (e) {}
        try {
          if (fullBtn)
            fullBtn.removeEventListener(
              "click",
              readerEventHandlers.onFullClick
            );
        } catch (e) {}
        try {
          if (bgBtn)
            bgBtn.removeEventListener("click", readerEventHandlers.onBgClick);
        } catch (e) {}
        try {
          if (zoomIn)
            zoomIn.removeEventListener("click", readerEventHandlers.onZoomIn);
        } catch (e) {}
        try {
          if (zoomOut)
            zoomOut.removeEventListener("click", readerEventHandlers.onZoomOut);
        } catch (e) {}
        try {
          if (toTop)
            toTop.removeEventListener("click", readerEventHandlers.onToTop);
        } catch (e) {}

        // 移除全局事件
        try {
          window.removeEventListener(
            "scroll",
            readerEventHandlers.updateCurrentPageInfo,
            { passive: true }
          );
        } catch (e) {}
        try {
          window.removeEventListener(
            "mousemove",
            readerEventHandlers.onMousemove,
            { passive: true }
          );
        } catch (e) {}
        try {
          window.removeEventListener("keydown", readerEventHandlers.onKeydown, {
            passive: true,
          });
        } catch (e) {}
        try {
          document.removeEventListener(
            "fullscreenchange",
            readerEventHandlers.onFullscreen
          );
        } catch (e) {}
      } catch (e) {
        // ignore
      }
      // 清理定时器与状态
      if (readerToolsTimer) {
        clearTimeout(readerToolsTimer);
        readerToolsTimer = null;
      }
      readerEventHandlers = {};
      readerEventsBound = false;
      unbindReaderEvents = null;
    };

    // 立即初始化阅读器 UI
    //  initReaderUI();
  }

  /*** 退出阅读模式 ***/
  function exitReader() {
    isReading = false;
    setAutoReaderStatus(false); // 关闭全局阅读器模式

    // 解绑阅读器事件
    try {
      if (typeof unbindReaderEvents === "function") unbindReaderEvents();
    } catch (e) {}

    // 退出全屏
    if (document.fullscreenElement) {
      document.exitFullscreen().catch((e) => {});
    }
    // 如果当前 URL 和进入时的 URL 不一致,说明用户切换了章节
    if (readerStartUrl && window.location.href !== readerStartUrl) {
      window.location.reload();
      return;
    }

    // 如果 URL 没变,说明还是原来那个帖子,直接恢复 DOM 快照(秒开,无需刷新)
    if (originalPageHTML) {
      console.log("🔄 恢复原始页面内容...");

      // 移除阅读器样式
      const readerStyle = document.getElementById("yamibo-reader-style");
      if (readerStyle) {
        readerStyle.remove();
        console.log("🧹 已移除阅读器样式");
      }

      const loaderStyle = document.getElementById("chapter-loader-style");
      if (loaderStyle) loaderStyle.remove();

      document.body.innerHTML = originalPageHTML;
      document.title = originalPageTitle;

      // 重新绑定入口按钮事件
      const restoredButton = document.getElementById("reader-toggle");
      if (restoredButton) {
        restoredButton.addEventListener("click", () => {
          if (!isReading) {
            enterReader();
            setAutoReaderStatus(true);
          } else {
            exitReader();
          }
        });
      }
      console.log("✅ 页面内容已恢复");
    } else {
      location.reload();
    }
  }
  // 无缝章节加载:加载提示与错误显示
  function showLoadingIndicator(message = "正在加载章节...") {
    try {
      // 移除已存在的加载提示
      const existing = document.getElementById("chapter-loader");
      if (existing) existing.remove();

      const loader = document.createElement("div");
      loader.id = "chapter-loader";
      loader.innerHTML = `
        <div class="loader-content">
          <div class="loader-spinner"></div>
          <div class="loader-text">${message}</div>
        </div>
      `;

      const style = document.createElement("style");
      style.id = "chapter-loader-style";
      style.textContent = `
        #chapter-loader { position: fixed; top:0; left:0; width:100%; height:100%; background: rgba(0,0,0,0.6); display:flex; align-items:center; justify-content:center; z-index:2147483648; }
        #chapter-loader .loader-content { color:#fff; padding:18px 24px; border-radius:10px; background: rgba(0,0,0,0.45); text-align:center; }
        #chapter-loader .loader-spinner { width:30px; height:30px; border:3px solid rgba(255,255,255,0.25); border-top:3px solid #fff; border-radius:50%; animation: yamibo-spin 1s linear infinite; margin:0 auto 8px }
        @keyframes yamibo-spin { from { transform: rotate(0deg); } to { transform: rotate(360deg); } }
        #chapter-loader .loader-text { font-size:14px; }
      `;

      document.head.appendChild(style);
      document.body.appendChild(loader);
    } catch (e) {
      console.warn("showLoadingIndicator error", e);
    }
  }

  function hideLoadingIndicator() {
    try {
      const loader = document.getElementById("chapter-loader");
      if (loader) {
        loader.remove();
      }
      const style = document.getElementById("chapter-loader-style");
      if (style) style.remove();
    } catch (e) {
      // ignore
    }
  }
// ====== 刷新和工具函数 (新添加) ======

  function manualRefresh() {
    // 1. 获取图片容器(ID 在 initReaderUI 中定义)
    const cwContainer = document.getElementById('cw-container');
    
    // 2. 检查全局图片列表 'images' 和容器是否存在
    // 假设 'images' 是全局变量,存储了当前章节的图片 URL 数组
    if (!cwContainer || !Array.isArray(images) || images.length === 0) {
        console.error("❌ 无法重新渲染图片:图片容器或图片列表为空。");
        showErrorMessage("图片列表为空或容器未找到。");
        return;
    }

    // 3. 调用图片渲染函数
    console.log("手动刷新:重新渲染图片...");
    showLoadingIndicator(); 
    
    // 假设 renderImages(containerElement, imageListArray)
    // 这是您脚本中实际渲染图片的函数
    renderImages(cwContainer, images); 
    
    // 滚动到顶部 (可选)
    window.scrollTo({ top: 0, behavior: "smooth" });

    // 延迟隐藏加载指示器,给用户一个反馈
    setTimeout(hideLoadingIndicator, 300);
}

  function setupRefreshButton() {
    // 1. 创建新的刷新按钮元素
    const refreshBtn = document.createElement('button');
    refreshBtn.id = 'cw-refresh';
    refreshBtn.dataset.tooltip = '手动刷新 (R)';
    refreshBtn.innerHTML = '↻';

    // 2. 将新按钮添加到 body (CSS 会负责定位)
    document.body.appendChild(refreshBtn);

    // 3. 绑定点击事件
    refreshBtn.addEventListener('click', manualRefresh);

    // 4. 绑定键盘快捷键 R (避免在输入框中触发)
    document.addEventListener("keydown", (e) => {
        if (e.key === "r" || e.key === "R") {
            // 确保焦点不在输入框
            if (e.target.tagName !== 'INPUT' && e.target.tagName !== 'TEXTAREA') {
                e.preventDefault(); 
                manualRefresh();
            }
        }
    });
 }

  function showErrorMessage(message) {
    try {
      const id = "chapter-error-msg";
      const existing = document.getElementById(id);
      if (existing) existing.remove();

      const el = document.createElement("div");
      el.id = id;
      el.style.cssText =
        "position:fixed;left:50%;top:40%;transform:translate(-50%,-50%);background:#ff6b6b;color:#fff;padding:12px  18px;border-radius:8px;z-index:2147483649;font-size:14px;box-shadow:0 8px 30px rgba(0,0,0,0.4)";
      el.textContent = message;
      document.body.appendChild(el);
      setTimeout(() => {
        try {
          el.style.opacity = "0";
          setTimeout(() => el.remove(), 250);
        } catch (e) {}
      }, 2500);
    } catch (e) {
      console.warn("showErrorMessage error", e);
    }
  }

  // 无缝章节切换
  async function loadNewChapter(chapterUrl, chapterTitle) {
    try {
      console.log("📖 开始无缝加载章节:", chapterTitle);
      showLoadingIndicator("正在加载章节...");

      const response = await fetch(chapterUrl, {
        method: "GET",
        credentials: "include",
        headers: {
          Accept:
            "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8",
          "Accept-Language": "zh-CN,zh;q=0.8,en-US;q=0.5,en;q=0.3",
          Referer: window.location.href,
          "User-Agent": navigator.userAgent,
        },
      });

      if (!response.ok) {
        throw new Error(`HTTP ${response.status}: ${response.statusText}`);
      }

      const html = await response.text();
      const parser = new DOMParser();
      const doc = parser.parseFromString(html, "text/html");

      // 提取图片 URL
      const newImages = collectImagesFromDocument(doc)
        .map((u) => {
          if (!u) return null;
          let url = String(u);
          if (url.startsWith("//")) url = "https:" + url;
          if (!/^https?:\/\//i.test(url))
            url = "https://bbs.yamibo.com/" + url.replace(/^\/+/, "");
          if (url.startsWith("http://"))
            url = url.replace(/^http:\/\//, "https://");
          url = url.replace(/^https?:\/\/https?:\/\//, "https://");
          return url;
        })
        .filter(Boolean);

      if (!newImages || newImages.length === 0) {
        hideLoadingIndicator();
        showErrorMessage("未找到漫画图片,已跳转到原帖");
        location.href = chapterUrl;
        return;
      }

      // 更新全局图片数组
      images = newImages;
      // 更新标题栏
      const titleDiv = document.getElementById("cw-title-bar");
      if (titleDiv)
        titleDiv.textContent =
          savedThreadTitle ||
          originalSeriesTitle ||
          document.title.split(" - ")[0] ||
          "漫画标题";
      // 尝试提取并保存新章节标题
      const newH1 = doc.querySelector("h1.ts");
      if (newH1) {
        const subjectSpan = newH1.querySelector(
          '#thread_subject, span[id^="thread_subject"]'
        );
        savedThreadTitle = subjectSpan
          ? subjectSpan.textContent.trim()
          : newH1.textContent.trim();
      } else {
        const newTitleEl = doc.querySelector(
          '#thread_subject, span[id^="thread_subject"]'
        );
        if (newTitleEl) savedThreadTitle = newTitleEl.textContent.trim();
      }

      // 更新浏览器地址栏(不刷新页面)
      try {
        window.history.pushState({}, "", chapterUrl);
      } catch (e) {
        /* ignore */
      }

      // 统一刷新UI(标题、图片、缩放、按钮等)
      initReaderUI();
      // 滚动到顶部
      window.scrollTo({ top: 0, behavior: "smooth" });

      hideLoadingIndicator();

      console.log("🎉 章节加载完成:", chapterTitle);
    } catch (error) {
      console.error("❌ 加载章节失败:", error);
      hideLoadingIndicator();
      showErrorMessage(
        "加载章节失败: " +
          (error && error.message ? error.message : String(error))
      );
      // 若失败,1s 后回退到原帖(避免卡死在阅读器)
      setTimeout(() => {
        try {
          location.href = chapterUrl;
        } catch (e) {}
      }, 1000);
    }
  }

  // 手动清理缓存功能
  if (typeof unsafeWindow !== "undefined") {
    unsafeWindow.clearAllCache = clearAllCache;
    unsafeWindow.viewCacheStatus = viewCacheStatus;
  } else {
    window.clearAllCache = clearAllCache;
    window.viewCacheStatus = viewCacheStatus;
  }
})();