// ==UserScript== // @name 智谱清言(ChatGLM) - 侧边栏会话跳转导航 | ChatGLM Sidebar Navigator // @namespace https://github.com/sakura11111111111111/ChatGLM-Sidebar-Jump-Axis // @version 1.0.0 // @description 为智谱清言(ChatGLM.cn)添加右侧悬浮侧边栏。核心功能:1. 双重视图:支持无缝切换"宝石连线"与"问题列表"模式;2. 布局锁死:修复内容溢出问题,按钮永远可见;3. 会话跳转:自动生成问题锚点,平滑滚动定位;4. 智能记忆:自动记录星标节点。 // @author zdm@Gai.cn // @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 // @downloadURL 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); })();