Greasy Fork

Greasy Fork is available in English.

智谱清言(ChatGLM) - 侧边栏会话跳转导航 | ChatGLM Sidebar Navigator

为智谱清言(ChatGLM.cn)添加右侧悬浮侧边栏。核心功能:1. 双重视图:支持无缝切换"宝石连线"与"问题列表"模式;2. 布局锁死:修复内容溢出问题,按钮永远可见;3. 会话跳转:自动生成问题锚点,平滑滚动定位;4. 智能记忆:自动记录星标节点。

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

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         智谱清言(ChatGLM) - 侧边栏会话跳转导航 | ChatGLM Sidebar Navigator
// @namespace    https://github.com/sakura11111111111111/ChatGLM-Sidebar-Jump-Axis
// @version      1.0.1
// @description  为智谱清言(ChatGLM.cn)添加右侧悬浮侧边栏。核心功能:1. 双重视图:支持无缝切换"宝石连线"与"问题列表"模式;2. 布局锁死:修复内容溢出问题,按钮永远可见;3. 会话跳转:自动生成问题锚点,平滑滚动定位;4. 智能记忆:自动记录星标节点。
// @author       [email protected]
// @match        https://chatglm.cn/*
// @icon         https://chatglm.cn/img/icons/favicon.svg
// @license      MIT
// @homepageURL  https://space.bilibili.com/497930349
// @supportURL   https://github.com/sakura11111111111111/ChatGLM-Sidebar-Jump-Axis/issues
// @grant        none
// ==/UserScript==

(function () {
  'use strict';

  // === 1. 宝石素材库 (请在此处填入 Base64 字符串) ===
  // 星标态 
     const GEM_STAR =""



  // 默认态 
     const GEM_NORMAL =""



  // === 2. 智能记忆系统 ===
  function getChatId(fallbackText) {
    const match = window.location.href.match(/\/(detail|share)\/([a-zA-Z0-9]+)/);
    if (match) return match[2];
    let fingerprint = document.title;
    if (fallbackText) fingerprint += "_" + fallbackText.replace(/\s/g, '').slice(0, 15);
    return "session_" + fingerprint;
  }
  function getStarredList(cid) {
    const raw = localStorage.getItem(`chatglm_stars_${cid}`);
    return raw ? JSON.parse(raw) : [];
  }
  function saveStarredList(cid, list) {
    localStorage.setItem(`chatglm_stars_${cid}`, JSON.stringify(list));
  }
  function toggleStar(qid, cid) {
    let list = getStarredList(cid);
    const idx = list.indexOf(qid);
    if (idx === -1) list.push(qid); else list.splice(idx, 1);
    saveStarredList(cid, list);
    return idx === -1;
  }

  // === 3. 样式定义 ===
  // 核心策略:[PART A] 完全复用 V13.2 的原始样式作为基准。 [PART B] 将列表模式作为独立扩展。
  const STYLES = `
        /* ========================================= */
        /* [PART A] 原始 V13.2 宝石模式核心样式 (严格复刻) */
        /* ========================================= */

        /* 高亮动画 */
        @keyframes glm-highlight-pulse {
            0% { box-shadow: 0 0 0 transparent; background-color: transparent; border-color: transparent; }
            30% { box-shadow: 0 0 25px rgba(64, 158, 255, 0.6); background-color: rgba(64, 158, 255, 0.1); border-color: rgba(64, 158, 255, 0.8); }
            100% { box-shadow: 0 0 0 transparent; background-color: transparent; border-color: transparent; }
        }
        .glm-flash-target {
            animation: glm-highlight-pulse 1.8s ease-out forwards;
            border: 1px solid transparent;
            border-radius: 8px;
        }

        /* === 外层包裹器 (Layout Fix) === */
        #glm-nav-wrapper {
            position: fixed;
            right: 30px;
            top: 50%;
            transform: translateY(-50%);
            max-height: 80vh;
            height: auto;
            display: flex;
            flex-direction: column;
            align-items: center;
            z-index: 99998;
            transition: transform 0.4s cubic-bezier(0.2, 0.8, 0.2, 1), right 0.4s ease, width 0.3s ease, background 0.3s, padding 0.3s;

            /* 默认状态 */
            width: 50px; padding: 0; border-radius: 0; background: transparent; border: none;
        }

        /* 折叠状态 */
        #glm-nav-wrapper.collapsed {
            transform: translateY(-50%) translateX(80px);
            right: 20px;
            pointer-events: none;
        }

        /* === 主内容区 === */
        #glm-nav-main-content {
            display: flex;
            flex-direction: column;
            align-items: center;
            width: 100%;
            flex: 1;
            min-height: 0;
            transition: opacity 0.3s;
        }

        #glm-nav-wrapper.collapsed #glm-nav-main-content {
            opacity: 0;
            pointer-events: none;
        }

        /* === 中间滚动区域 === */
        #glm-scroll-area {
            flex: 1;
            min-height: 0;
            width: 100%;
            overflow-y: auto;
            overflow-x: visible;
            padding: 5px 15px;
            scrollbar-width: none;
            -ms-overflow-style: none;
            display: flex;
            flex-direction: column;
            position: relative;
        }
        #glm-scroll-area::-webkit-scrollbar { display: none; }

        /* === 宝石节点 (V13.2 原版定义) === */
        .glm-nav-dot {
            width: 24px; height: 24px;
            background-size: contain; background-repeat: no-repeat; background-position: center;
            background-color: transparent;
            box-shadow: none;
            border: 1px solid rgba(255,255,255,0.1);
            border-radius: 50%;
            opacity: 0.6; filter: grayscale(40%);
            transform: scale(0.85); cursor: pointer;
            transition: all 0.4s cubic-bezier(0.34, 1.56, 0.64, 1);
            position: relative; flex-shrink: 0;
            margin: 18px auto;
            overflow: visible;
            z-index: 2;
        }

        /* === 连线 (V13.2 原版定义) === */
        .glm-nav-dot::after {
            content: ''; position: absolute; left: 50%; transform: translateX(-50%);
            top: 100%; height: 38px; width: 1.5px;
            background: rgba(255, 255, 255, 0.15);
            pointer-events: none; z-index: -1; transition: all 0.4s;
        }
        .glm-nav-dot:last-child::after { display: none; }

        /* === 悬浮/激活/星标 特效 (V13.2 原版定义 - 绝对保留) === */
        .glm-nav-dot:hover { opacity: 1; filter: grayscale(0%); transform: scale(1.1); border-color: rgba(255,255,255,0.3); }
        .glm-nav-dot.active { opacity: 1; filter: grayscale(0%) drop-shadow(0 0 8px rgba(64, 158, 255, 0.6)); transform: scale(1.3); z-index: 10; border: none; }
        .glm-nav-dot.is-starred { opacity: 1 !important; filter: grayscale(0%) brightness(1.1) !important; transform: scale(1.1); border: none; }
        .glm-nav-dot.is-starred.active { transform: scale(1.4); filter: drop-shadow(0 0 10px rgba(255, 60, 60, 0.8)) !important; }

        /* === 按钮通用 === */
        .glm-elevator-btn {
            width: 28px; height: 28px; flex-shrink: 0;
            background: rgba(30, 30, 35, 0.85); backdrop-filter: blur(5px);
            border: 1px solid rgba(255, 255, 255, 0.2); border-radius: 8px;
            color: rgba(255, 255, 255, 0.7);
            display: flex; align-items: center; justify-content: center;
            cursor: pointer; font-size: 12px; transition: all 0.2s;
            user-select: none; margin: 2px 0; z-index: 99999;
        }
        .glm-elevator-btn:hover {
            background: rgba(64, 158, 255, 0.2); border-color: rgba(64, 158, 255, 0.6); color: #fff; transform: scale(1.1);
        }
        #glm-toggle-btn {
            position: absolute; bottom: -45px; width: 24px; height: 24px;
            background: rgba(20, 20, 20, 0.6); backdrop-filter: blur(4px);
            border: 1px solid rgba(255, 255, 255, 0.1); border-radius: 50%;
            color: rgba(255, 255, 255, 0.6); display: flex; align-items: center; justify-content: center;
            cursor: pointer; font-size: 12px; transition: all 0.3s;
            z-index: 100000; pointer-events: auto;
        }
        #glm-toggle-btn:hover { background: rgba(64, 158, 255, 0.8); color: #fff; transform: scale(1.2); }
        #glm-nav-wrapper.collapsed #glm-toggle-btn {
            transform: translateX(-70px) translateY(-50%); bottom: 50%;
            width: 30px; height: 60px; border-radius: 15px 0 0 15px;
            background: rgba(64, 158, 255, 0.3); box-shadow: -2px 0 10px rgba(0,0,0,0.2);
        }
        #glm-nav-wrapper.collapsed #glm-toggle-btn:hover { background: rgba(64, 158, 255, 0.9); width: 35px; }

        /* 提示框 */
        #glm-global-tooltip {
            position: fixed; background: rgba(15, 15, 20, 0.95); backdrop-filter: blur(10px);
            color: rgba(255, 255, 255, 0.95); padding: 10px 16px; font-size: 13px; font-weight: 500;
            border-radius: 8px; border: 1px solid rgba(255,255,255,0.1);
            box-shadow: 0 10px 30px rgba(0,0,0,0.6); z-index: 99999; pointer-events: none;
            opacity: 0; visibility: hidden; transition: opacity 0.2s, transform 0.2s;
            transform: translateY(-50%) translateX(15px);
            font-family: -apple-system, BlinkMacSystemFont, sans-serif;
            max-width: 400px; white-space: normal; line-height: 1.5;
        }
        #glm-global-tooltip.visible { opacity: 1; visibility: visible; transform: translateY(-50%) translateX(0); }
        #glm-global-tooltip::before {
            content: ''; position: absolute; left: -5px; top: 50%; transform: translateY(-50%) rotate(45deg);
            width: 10px; height: 10px; background: inherit;
            border-left: 1px solid rgba(255,255,255,0.1); border-bottom: 1px solid rgba(255,255,255,0.1); z-index: -1;
        }


        /* ========================================= */
        /* [PART B] 扩展:列表模式 (List Mode) */
        /* 只有当 #glm-nav-wrapper 拥有 .list-mode 类时才生效 */
        /* ========================================= */

        /* 视图切换按钮 */
        #glm-btn-view { font-size: 14px; margin-bottom: 6px; }

        /* 列表模式容器 */
        #glm-nav-wrapper.list-mode {
            width: 260px; padding: 15px 10px; align-items: stretch;
            background: rgba(18, 18, 24, 0.92); backdrop-filter: blur(12px);
            border-radius: 12px; border: 1px solid rgba(255, 255, 255, 0.08);
            box-shadow: 0 8px 32px rgba(0, 0, 0, 0.5);
        }

        /* 列表模式滚动区 */
        #glm-nav-wrapper.list-mode #glm-scroll-area {
            overflow-x: hidden; padding: 5px 0;
        }

        /* 列表模式下的节点重置 (抹除宝石外观) */
        #glm-nav-wrapper.list-mode .glm-nav-dot {
            background-image: none !important;
            width: auto; height: auto; border-radius: 6px;
            margin: 4px 5px; padding: 8px 10px 8px 20px;
            border: none; transform: none !important;
            opacity: 0.7; filter: none; background-color: transparent;
            display: flex; align-items: center;
        }

        /* 列表模式悬浮 */
        #glm-nav-wrapper.list-mode .glm-nav-dot:hover { opacity: 1; background-color: rgba(255, 255, 255, 0.08); }

        /* 列表模式激活 (矩形背景仅在此处生效) */
        #glm-nav-wrapper.list-mode .glm-nav-dot.active {
            opacity: 1; background-color: rgba(64, 158, 255, 0.15); color: #fff;
        }

        /* 列表模式隐藏连线 */
        #glm-nav-wrapper.list-mode .glm-nav-dot::after { display: none; }

        /* 列表模式文字 */
        .glm-nav-label { display: none; } /* 默认隐藏 */
        #glm-nav-wrapper.list-mode .glm-nav-label {
            display: block; color: rgba(255, 255, 255, 0.85);
            font-size: 13px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
            width: 100%; text-align: left;
        }
        #glm-nav-wrapper.list-mode .glm-nav-dot.active .glm-nav-label { color: #fff; font-weight: 500; }
        #glm-nav-wrapper.list-mode .glm-nav-dot.is-starred .glm-nav-label { color: #FFD700; }

        /* 列表模式小圆点 */
        #glm-nav-wrapper.list-mode .glm-nav-dot::before {
            content: ''; position: absolute; left: 8px; top: 50%; transform: translateY(-50%);
            width: 4px; height: 4px; border-radius: 50%; background-color: rgba(255, 255, 255, 0.4);
        }
        #glm-nav-wrapper.list-mode .glm-nav-dot.active::before {
            background-color: #409EFF; width: 6px; height: 6px; left: 7px;
        }
        #glm-nav-wrapper.list-mode .glm-nav-dot.is-starred::before { background-color: #FFD700; }
    `;

  const styleSheet = document.createElement("style");
  styleSheet.innerText = STYLES;
  document.head.appendChild(styleSheet);

  // === 4. DOM 结构 ===
  const wrapper = document.createElement('div');
  wrapper.id = 'glm-nav-wrapper';
  document.body.appendChild(wrapper);

  const mainContent = document.createElement('div');
  mainContent.id = 'glm-nav-main-content';
  wrapper.appendChild(mainContent);

  // 0. 视图切换按钮
  const btnView = document.createElement('div');
  btnView.id = 'glm-btn-view';
  btnView.className = 'glm-elevator-btn';
  btnView.innerHTML = '≡';
  btnView.title = "切换列表视图";
  btnView.style.display = 'none';
  mainContent.appendChild(btnView);

  // 1. 顶部按钮
  const btnTop = document.createElement('div');
  btnTop.className = 'glm-elevator-btn';
  btnTop.innerHTML = '▲';
  btnTop.title = "回到顶部";
  btnTop.style.display = 'none';
  mainContent.appendChild(btnTop);

  // 2. 滚动区域
  const scrollArea = document.createElement('div');
  scrollArea.id = 'glm-scroll-area';
  mainContent.appendChild(scrollArea);

  // 3. 底部按钮
  const btnBottom = document.createElement('div');
  btnBottom.className = 'glm-elevator-btn';
  btnBottom.innerHTML = '▼';
  btnBottom.title = "直达最新";
  btnBottom.style.display = 'none';
  mainContent.appendChild(btnBottom);

  // 4. 折叠按钮
  const toggleBtn = document.createElement('div');
  toggleBtn.id = 'glm-toggle-btn';
  toggleBtn.innerHTML = '»';
  toggleBtn.title = "折叠/展开";
  wrapper.appendChild(toggleBtn);

  const tooltip = document.createElement('div');
  tooltip.id = 'glm-global-tooltip';
  document.body.appendChild(tooltip);

  let lastRenderedSignature = "";
  let isClickScrolling = false;
  let currentQuestions = [];

  // === 5. 状态管理 ===
  let isCollapsed = false;
  let isListMode = false;

  // 视图切换逻辑
  function toggleListMode() {
    isListMode = !isListMode;
    if (isListMode) {
      wrapper.classList.add('list-mode');
      btnView.innerHTML = '×';
      btnView.title = "关闭列表";
      setTimeout(() => {
        const activeDot = scrollArea.querySelector('.glm-nav-dot.active');
        if (activeDot) activeDot.scrollIntoView({ block: 'center', behavior: 'auto' });
      }, 50);
    } else {
      wrapper.classList.remove('list-mode');
      btnView.innerHTML = '≡';
      btnView.title = "切换列表视图";
      setTimeout(() => {
        const activeDot = scrollArea.querySelector('.glm-nav-dot.active');
        if (activeDot) activeDot.scrollIntoView({ block: 'center', behavior: 'auto' });
      }, 50);
    }
  }
  btnView.onclick = (e) => { e.stopPropagation(); toggleListMode(); };

  // 折叠逻辑
  function toggleSidebar(forceState = null) {
    if (forceState !== null) isCollapsed = forceState;
    else isCollapsed = !isCollapsed;

    if (isCollapsed) {
      wrapper.classList.add('collapsed');
      toggleBtn.innerHTML = '💎';
      tooltip.classList.remove('visible');
    } else {
      wrapper.classList.remove('collapsed');
      toggleBtn.innerHTML = '»';
    }
  }
  toggleBtn.onclick = (e) => { e.stopPropagation(); toggleSidebar(); };

  function checkResponsive() {
    if (window.innerWidth < 1400) toggleSidebar(true);
  }
  checkResponsive();
  window.addEventListener('resize', () => setTimeout(checkResponsive, 200));

  // === 6. 渲染逻辑 ===
  const observerOptions = { root: null, rootMargin: '-45% 0px -45% 0px', threshold: 0 };
  const scrollObserver = new IntersectionObserver((entries) => {
    if (isClickScrolling) return;
    entries.forEach(entry => { if (entry.isIntersecting) activateDot(entry.target.id); });
  }, observerOptions);

  function activateDot(targetId) {
    const allDots = scrollArea.querySelectorAll('.glm-nav-dot');
    let activeDot = null;
    allDots.forEach(dot => {
      if (dot.dataset.targetId === targetId) {
        dot.classList.add('active'); activeDot = dot;
      } else {
        dot.classList.remove('active');
      }
    });
    if (activeDot) {
      const containerHeight = scrollArea.clientHeight;
      const dotTop = activeDot.offsetTop;
      const dotHeight = activeDot.clientHeight;
      scrollArea.scrollTo({ top: dotTop - (containerHeight / 2) + (dotHeight / 2), behavior: 'smooth' });
    }
  }

  btnTop.onclick = () => {
    if (currentQuestions.length > 0) {
      isClickScrolling = true;
      const target = currentQuestions[0];
      target.scrollIntoView({ behavior: 'smooth', block: 'start' });
      activateDot(target.id);
      setTimeout(() => isClickScrolling = false, 1000);
    }
  };
  btnBottom.onclick = () => {
    if (currentQuestions.length > 0) {
      isClickScrolling = true;
      const target = currentQuestions[currentQuestions.length - 1];
      target.scrollIntoView({ behavior: 'smooth', block: 'start' });
      activateDot(target.id);
      setTimeout(() => isClickScrolling = false, 1000);
    }
  };

  function generateNavNodes() {
    const allQuestions = document.querySelectorAll('[id^="row-question-"]');
    const validQuestions = Array.from(allQuestions).filter(q => /^row-question-\d+$/.test(q.id) && q.offsetHeight > 0);
    currentQuestions = validQuestions;

    const hasContent = validQuestions.length > 0;
    const showElevator = validQuestions.length > 3;

    wrapper.style.display = hasContent ? 'flex' : 'none';
    btnView.style.display = hasContent ? 'flex' : 'none';
    btnTop.style.display = showElevator ? 'flex' : 'none';
    btnBottom.style.display = showElevator ? 'flex' : 'none';

    if (!hasContent) return;

    const firstQuestionText = validQuestions[0].innerText;
    const currentChatId = getChatId(firstQuestionText);
    const currentSignature = currentChatId + "|" + validQuestions.map(q => q.id).join('|');

    if (currentSignature === lastRenderedSignature) return;
    lastRenderedSignature = currentSignature;

    scrollArea.innerHTML = '';
    scrollObserver.disconnect();

    const starredList = getStarredList(currentChatId);

    validQuestions.forEach((q, index) => {
      scrollObserver.observe(q);
      const dot = document.createElement('div');
      dot.className = 'glm-nav-dot';
      dot.dataset.targetId = q.id;

      const isStarred = starredList.includes(q.id);
      if (isStarred && GEM_STAR) dot.style.backgroundImage = `url(${GEM_STAR})`;
      else if (!isStarred && GEM_NORMAL) dot.style.backgroundImage = `url(${GEM_NORMAL})`;
      if (isStarred) dot.classList.add('is-starred');

      let textRaw = (q.querySelector('.question-txt') || q).innerText;
      const cleanText = textRaw.replace(/\s+/g, ' ').trim();
      const tooltipText = `Q${index + 1}: ${cleanText.slice(0, 80)}${cleanText.length > 80 ? '...' : ''}`;
      const labelText = cleanText.slice(0, 60);

      dot.dataset.rawText = tooltipText;

      // 内部文本标签 (列表模式专用)
      const labelSpan = document.createElement('span');
      labelSpan.className = 'glm-nav-label';
      labelSpan.innerText = labelText;
      dot.appendChild(labelSpan);

      dot.onmouseenter = () => {
        if (isCollapsed) return;
        if (isListMode) return; // 列表模式不需要 Tooltip

        const rect = dot.getBoundingClientRect();
        tooltip.innerText = (dot.classList.contains('is-starred') ? "⭐ " : "") + dot.dataset.rawText;
        tooltip.style.right = (window.innerWidth - rect.left + 25) + 'px';
        tooltip.style.top = (rect.top + rect.height / 2) + 'px';
        tooltip.classList.add('visible');
      };
      dot.onmouseleave = () => tooltip.classList.remove('visible');

      dot.onclick = (e) => {
        e.stopPropagation();
        isClickScrolling = true;
        activateDot(q.id);
        q.scrollIntoView({ behavior: 'smooth', block: 'center' });
        q.classList.remove('glm-flash-target');
        void q.offsetWidth;
        q.classList.add('glm-flash-target');
        setTimeout(() => { isClickScrolling = false; }, 1000);
      };

      dot.ondblclick = (e) => {
        e.stopPropagation();
        const nowStarred = toggleStar(q.id, currentChatId);
        if (nowStarred && GEM_STAR) dot.style.backgroundImage = `url(${GEM_STAR})`;
        else if (!nowStarred && GEM_NORMAL) dot.style.backgroundImage = `url(${GEM_NORMAL})`;
        nowStarred ? dot.classList.add('is-starred') : dot.classList.remove('is-starred');

        if (!isListMode) {
          tooltip.innerText = (nowStarred ? "⭐ " : "") + dot.dataset.rawText;
          dot.style.transform = "scale(1.6)";
          setTimeout(() => dot.style.transform = "", 200);
        }
      };

      scrollArea.appendChild(dot);
    });
  }

  let timeout = null;
  const observer = new MutationObserver(() => {
    if (timeout) clearTimeout(timeout);
    timeout = setTimeout(generateNavNodes, 500);
  });
  observer.observe(document.body, { childList: true, subtree: true });
  setTimeout(generateNavNodes, 1000);

})();