// ==UserScript== // @name AI 宝石导航 (V14.21 豆包显色修复版) // @namespace https://github.com/sakura11111111111111/ChatGLM-Sidebar-Jump-Axis // @version 3.1.0 // @description [V14.21] 基于 V14.20 严格修正。仅修改连线颜色逻辑,完美解决豆包连线不可见问题,绝不影响其他站点。 // @author zdm@Gai.cn // @match https://chatglm.cn/* // @match https://chat.deepseek.com/* // @match https://chatgpt.com/* // @match https://grok.com/* // @match https://x.com/* // @match https://www.doubao.com/* // @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 https://update.greasyfork.icu/scripts/556555/AI%20%E5%AE%9D%E7%9F%B3%E5%AF%BC%E8%88%AA%20%28V1421%20%E8%B1%86%E5%8C%85%E6%98%BE%E8%89%B2%E4%BF%AE%E5%A4%8D%E7%89%88%29.user.js // @updateURL https://update.greasyfork.icu/scripts/556555/AI%20%E5%AE%9D%E7%9F%B3%E5%AF%BC%E8%88%AA%20%28V1421%20%E8%B1%86%E5%8C%85%E6%98%BE%E8%89%B2%E4%BF%AE%E5%A4%8D%E7%89%88%29.meta.js // ==/UserScript== (function() { 'use strict'; // === 1. 默认素材库 === const DEFAULT_GEM_STAR ="" const DEFAULT_GEM_NORMAL ="" // ========================================================================== // [MODULE 1] 适配器:配置层 (保持 V14.20 原样) // ========================================================================== const ADAPTERS = [ { name: "ChatGLM", match: "chatglm.cn", getQuestions: () => { const all = document.querySelectorAll('[id^="row-question-"]'); return Array.from(all).filter(q => /^row-question-\d+$/.test(q.id) && q.offsetHeight > 0); }, getText: (el) => (el.querySelector('.question-txt') || el).innerText, getChatId: (firstQText) => { const match = window.location.href.match(/\/(detail|share)\/([a-zA-Z0-9]+)/); return match ? match[2] : "session_" + document.title; } }, { name: "DeepSeek", match: "chat.deepseek.com", getQuestions: () => { const allMsgs = document.querySelectorAll('.ds-message'); const valid = Array.from(allMsgs).filter(el => { let parent = el.parentElement; for(let i=0; i<3; i++) { if(!parent) break; const style = window.getComputedStyle(parent); if ((style.display === 'flex' && (style.alignItems === 'flex-end' || style.justifyContent === 'flex-end')) || style.flexDirection === 'row-reverse') { return true; } parent = parent.parentElement; } return false; }); valid.forEach((el, index) => { if (!el.id) el.id = `gem_deepseek_id_${index}`; }); return valid; }, getText: (el) => el.textContent.trim(), getChatId: (firstQText) => { const match = window.location.href.match(/chat\/([a-zA-Z0-9\-]+)/); return match ? match[1] : "ds_session_" + firstQText.slice(0, 10); } }, { name: "ChatGPT", match: "chatgpt.com", getQuestions: () => { const all = document.querySelectorAll('div[data-message-author-role="user"]'); const valid = Array.from(all); valid.forEach((el, index) => { if (!el.id) el.id = `gem_chatgpt_id_${index}`; }); return valid; }, getText: (el) => { const textNode = el.querySelector('.whitespace-pre-wrap'); return textNode ? textNode.innerText : el.innerText; }, getChatId: (firstQText) => { const match = window.location.href.match(/\/c\/([a-zA-Z0-9\-]+)/); return match ? match[1] : "gpt_nav_" + (firstQText ? firstQText.slice(0, 10) : "new"); }, quarantineClass: "site-chatgpt" }, { name: "Grok", match: ["grok.com", "x.com"], getQuestions: () => { const allBubbles = document.querySelectorAll('.message-bubble'); const valid = []; allBubbles.forEach(el => { const parent = el.parentElement; if (!parent) return; if (parent.className.includes('items-end') || window.getComputedStyle(parent).alignItems === 'flex-end') { valid.push(el); } }); valid.forEach((el, index) => { if (!el.id || !el.id.startsWith('gem_grok_id_')) el.id = `gem_grok_id_${index}`; }); return valid; }, getText: (el) => { const content = el.querySelector('.response-content-markdown'); return content ? content.innerText : el.innerText; }, getChatId: (firstQText) => { const match = window.location.href.match(/(grok|chat)\/([a-zA-Z0-9\-]+)/); if (match) return match[2]; return "grok_sess_" + (firstQText ? firstQText.slice(0, 10).replace(/\s/g, '') : "new"); }, quarantineClass: "site-grok" }, { // 豆包适配器 (保持 V14.20 逻辑) name: "Doubao", match: ["doubao.com"], getQuestions: () => { const all = document.querySelectorAll('div[data-testid="message_content"]'); const valid = Array.from(all).filter(el => el.offsetHeight > 0 && el.classList.contains('justify-end')); valid.forEach((el, index) => { if (!el.id) el.id = `gem_doubao_${index}`; }); return valid; }, getText: (el) => { const textNode = el.querySelector('[data-testid="message_text_content"]'); return textNode ? textNode.innerText : "Message"; }, getChatId: (firstQText) => { const match = window.location.href.match(/chat\/([a-zA-Z0-9\-]+)/); return match ? match[1] : "doubao_" + (firstQText ? firstQText.slice(0, 10) : "default"); }, quarantineClass: "site-doubao", forceScrollMode: "auto", forceScrollBlock: "start" } ]; // ========================================================================== // [MODULE 2] 核心引擎 (保持 V14.20 原样) // ========================================================================== const currentAdapter = ADAPTERS.find(a => { if (Array.isArray(a.match)) { return a.match.some(m => window.location.href.includes(m)); } return window.location.href.includes(a.match); }); if (!currentAdapter) return; if (currentAdapter.quarantineClass) { document.body.classList.add(currentAdapter.quarantineClass); } function getStarredList(cid) { const raw = localStorage.getItem(`gem_nav_stars_${cid}`); return raw ? JSON.parse(raw) : []; } function saveStarredList(cid, list) { localStorage.setItem(`gem_nav_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; } function getUserSettings() { const raw = localStorage.getItem('gem_nav_settings'); return raw ? JSON.parse(raw) : { normalIcon: null, starIcon: null }; } function saveUserSettings(settings) { localStorage.setItem('gem_nav_settings', JSON.stringify(settings)); } function compressAndSaveImage(file, type, callback) { const reader = new FileReader(); reader.onload = (e) => { const img = new Image(); img.onload = () => { const canvas = document.createElement('canvas'); const ctx = canvas.getContext('2d'); canvas.width = 100; canvas.height = 100; ctx.drawImage(img, 0, 0, 100, 100); const base64 = canvas.toDataURL('image/png'); const settings = getUserSettings(); settings[type] = base64; saveUserSettings(settings); callback(base64); }; img.src = e.target.result; }; reader.readAsDataURL(file); } // 样式定义 const STYLES = ` @keyframes glm-highlight-pulse { 0% { box-shadow: 0 0 0 transparent; background-color: transparent; outline-color: transparent; } 30% { box-shadow: 0 0 25px rgba(64, 158, 255, 0.6); background-color: rgba(64, 158, 255, 0.1); outline-color: rgba(64, 158, 255, 0.8); } 100% { box-shadow: 0 0 0 transparent; background-color: transparent; outline-color: transparent; } } .glm-flash-target { animation: glm-highlight-pulse 1.8s ease-out forwards !important; border: 1px solid transparent; outline: 2px solid transparent; border-radius: 8px; } #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: 2147483647; transition: all 0.4s cubic-bezier(0.2, 0.8, 0.2, 1); width: 50px; padding: 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; } .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; } .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; } .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: 2147483647; pointer-events: none; opacity: 0; visibility: hidden; transition: opacity 0.2s, transform 0.2s; transform: translateY(-50%) translateX(15px); font-family: -apple-system, sans-serif; max-width: 400px; } #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; } #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; } #glm-settings-modal-overlay { position: fixed; top: 0; left: 0; width: 100vw; height: 100vh; background: rgba(0, 0, 0, 0.5); backdrop-filter: blur(5px); z-index: 200000; display: flex; justify-content: center; align-items: center; opacity: 0; visibility: hidden; transition: all 0.3s; } #glm-settings-modal-overlay.open { opacity: 1; visibility: visible; } #glm-settings-panel { width: 320px; background: rgba(25, 25, 30, 0.95); border: 1px solid rgba(255,255,255,0.1); border-radius: 16px; padding: 20px; box-shadow: 0 20px 50px rgba(0,0,0,0.6); transform: scale(0.9); transition: transform 0.3s; } #glm-settings-modal-overlay.open #glm-settings-panel { transform: scale(1); } .glm-st-header { font-size: 16px; font-weight: bold; color: #fff; margin-bottom: 20px; text-align: center; } .glm-st-row { margin-bottom: 20px; display: flex; align-items: center; justify-content: space-between; } .glm-st-label { color: rgba(255,255,255,0.7); font-size: 13px; } .glm-st-preview { width: 40px; height: 40px; border-radius: 50%; border: 1px solid rgba(255,255,255,0.2); background-size: contain; background-repeat: no-repeat; background-position: center; margin: 0 10px; } .glm-st-actions { display: flex; gap: 8px; } .glm-btn-upload { position: relative; overflow: hidden; display: inline-block; cursor: pointer; background: rgba(64, 158, 255, 0.2); color: #409EFF; padding: 4px 10px; border-radius: 4px; font-size: 12px; border: 1px solid rgba(64, 158, 255, 0.4); } .glm-btn-upload input[type=file] { position: absolute; top: 0; right: 0; opacity: 0; cursor: pointer; height: 100%; width: 100%; } .glm-btn-reset { cursor: pointer; background: rgba(255, 255, 255, 0.1); color: rgba(255,255,255,0.6); padding: 4px 10px; border-radius: 4px; font-size: 12px; border: 1px solid rgba(255,255,255,0.1); } .glm-btn-reset:hover { background: rgba(255, 80, 80, 0.2); color: #ff5050; border-color: rgba(255, 80, 80, 0.4); } #glm-btn-close-st { width: 100%; padding: 8px; background: #409EFF; color: white; border: none; border-radius: 8px; font-size: 14px; cursor: pointer; margin-top: 10px; } #glm-btn-close-st:hover { background: #66b1ff; } /* --- [V14.16] Grok 增强补丁:强制隔离,保护ChatGLM --- */ .site-chatgpt:not(.list-mode) .glm-nav-dot, .site-grok:not(.list-mode) .glm-nav-dot, .site-doubao:not(.list-mode) .glm-nav-dot { box-sizing: content-box !important; width: 24px !important; height: 24px !important; min-width: 24px !important; min-height: 24px !important; flex-shrink: 0 !important; border: 1px solid rgba(255,255,255,0.1) !important; border-radius: 50% !important; transform-origin: center center !important; } .site-chatgpt:not(.list-mode) .glm-nav-dot.active, .site-grok:not(.list-mode) .glm-nav-dot.active, .site-doubao:not(.list-mode) .glm-nav-dot.active { border: none !important; box-shadow: none !important; filter: drop-shadow(0 0 6px rgba(64, 158, 255, 0.9)) !important; transform: scale(1.3) !important; background-color: transparent !important; } .site-chatgpt:not(.list-mode) .glm-nav-dot.is-starred.active, .site-grok:not(.list-mode) .glm-nav-dot.is-starred.active, .site-doubao:not(.list-mode) .glm-nav-dot.is-starred.active { filter: drop-shadow(0 0 6px rgba(255, 60, 60, 0.9)) !important; } .site-chatgpt.list-mode, .site-grok.list-mode, .site-doubao.list-mode { width: 260px !important; align-items: stretch !important; } .site-chatgpt.list-mode .glm-nav-dot, .site-grok.list-mode .glm-nav-dot, .site-doubao.list-mode .glm-nav-dot { width: auto !important; height: auto !important; min-width: auto !important; min-height: auto !important; border-radius: 6px !important; padding: 8px 10px 8px 20px !important; justify-content: flex-start !important; display: flex !important; flex-direction: row !important; flex-shrink: 1 !important; } .site-chatgpt.list-mode .glm-nav-label, .site-grok.list-mode .glm-nav-label, .site-doubao.list-mode .glm-nav-label { display: block !important; text-align: left !important; width: 100% !important; position: static !important; } .site-chatgpt .glm-elevator-btn, .site-chatgpt #glm-toggle-btn, .site-grok .glm-elevator-btn, .site-grok #glm-toggle-btn, .site-doubao .glm-elevator-btn, .site-doubao #glm-toggle-btn { box-sizing: border-box !important; flex-shrink: 0 !important; min-height: 28px !important; min-width: 28px !important; display: flex !important; align-items: center !important; justify-content: center !important; } /* [V14.20 豆包可见性修复] 在白底上强制显示深色边框和背景 */ .site-doubao:not(.list-mode) .glm-nav-dot { border: 2px solid rgba(0, 0, 0, 0.3) !important; background-color: rgba(0, 0, 0, 0.05) !important; } .site-doubao:not(.list-mode) .glm-nav-dot.active { border: 2px solid #0057ff !important; background-color: rgba(0, 87, 255, 0.15) !important; } .site-doubao .glm-elevator-btn, .site-doubao #glm-toggle-btn { background: rgba(0,0,0,0.7) !important; color: white !important; } /* [V14.21] 修复豆包列表视图文字不清 */ .site-doubao.list-mode { background: rgba(240, 240, 240, 0.98) !important; /* 浅色背景 */ border: 1px solid #ccc !important; } .site-doubao.list-mode .glm-nav-label { color: #333 !important; /* 深色文字 */ } .site-doubao.list-mode .glm-nav-dot.active .glm-nav-label { color: #0057ff !important; font-weight: bold !important; } /* 气泡高亮适配 */ .site-chatgpt .glm-flash-target, .site-grok .glm-flash-target, .site-doubao .glm-flash-target { box-shadow: inset 0 0 20px rgba(64, 158, 255, 0.4), 0 0 25px rgba(64, 158, 255, 0.6) !important; position: relative !important; z-index: 10 !important; border-radius: var(--glm-flash-radius, 16px) !important; } `; const styleSheet = document.createElement("style"); styleSheet.innerText = STYLES; document.head.appendChild(styleSheet); // DOM 结构 const wrapper = document.createElement('div'); wrapper.id = 'glm-nav-wrapper'; // 身份标记 const href = window.location.href; if (href.includes('chatgpt.com')) { wrapper.classList.add('site-chatgpt'); } else if (href.includes('grok.com') || href.includes('x.com')) { wrapper.classList.add('site-grok'); } else if (href.includes('doubao.com')) { wrapper.classList.add('site-doubao'); // [V14.20] 注入豆包类名 } document.body.appendChild(wrapper); const mainContent = document.createElement('div'); mainContent.id = 'glm-nav-main-content'; wrapper.appendChild(mainContent); const btnSettings = document.createElement('div'); btnSettings.className = 'glm-elevator-btn'; btnSettings.innerHTML = '⚙️'; btnSettings.title = "外观设置"; btnSettings.style.marginBottom = '6px'; mainContent.appendChild(btnSettings); 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); const btnTop = document.createElement('div'); btnTop.className = 'glm-elevator-btn'; btnTop.innerHTML = '▲'; btnTop.title = "回到顶部"; btnTop.style.display = 'none'; mainContent.appendChild(btnTop); const scrollArea = document.createElement('div'); scrollArea.id = 'glm-scroll-area'; mainContent.appendChild(scrollArea); const btnBottom = document.createElement('div'); btnBottom.className = 'glm-elevator-btn'; btnBottom.innerHTML = '▼'; btnBottom.title = "直达最新"; btnBottom.style.display = 'none'; mainContent.appendChild(btnBottom); 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); const settingsOverlay = document.createElement('div'); settingsOverlay.id = 'glm-settings-modal-overlay'; settingsOverlay.innerHTML = `
💎 侧边栏外观设置
普通状态
默认
星标状态
默认
`; document.body.appendChild(settingsOverlay); // 逻辑控制 let lastRenderedSignature = ""; let isClickScrolling = false; let scrollDebounceTimer = null; // [V14.15] 滚动防抖计时器 let currentQuestions = []; let isCollapsed = false; let isListMode = false; function getCurrentIcons() { const settings = getUserSettings(); // 如果没有设置,且默认常量也为空,为了防止透明,这里不返回 url(...),在CSS里处理,或者前端逻辑保证 return { normal: settings.normalIcon || DEFAULT_GEM_NORMAL, star: settings.starIcon || DEFAULT_GEM_STAR }; } const previewNormal = settingsOverlay.querySelector('#glm-preview-normal'); const previewStar = settingsOverlay.querySelector('#glm-preview-star'); btnSettings.onclick = (e) => { e.stopPropagation(); const icons = getCurrentIcons(); previewNormal.style.backgroundImage = `url('${icons.normal}')`; previewStar.style.backgroundImage = `url('${icons.star}')`; settingsOverlay.classList.add('open'); }; settingsOverlay.querySelector('#glm-btn-close-st').onclick = () => settingsOverlay.classList.remove('open'); settingsOverlay.onclick = (e) => { if(e.target === settingsOverlay) settingsOverlay.classList.remove('open'); }; settingsOverlay.querySelector('#glm-upload-normal').onchange = function() { if (this.files && this.files[0]) compressAndSaveImage(this.files[0], 'normalIcon', (base64) => { previewNormal.style.backgroundImage = `url('${base64}')`; lastRenderedSignature = ""; generateNavNodes(); }); }; settingsOverlay.querySelector('#glm-reset-normal').onclick = () => { const s = getUserSettings(); s.normalIcon = null; saveUserSettings(s); previewNormal.style.backgroundImage = `url('${DEFAULT_GEM_NORMAL}')`; lastRenderedSignature = ""; generateNavNodes(); }; settingsOverlay.querySelector('#glm-upload-star').onchange = function() { if (this.files && this.files[0]) compressAndSaveImage(this.files[0], 'starIcon', (base64) => { previewStar.style.backgroundImage = `url('${base64}')`; lastRenderedSignature = ""; generateNavNodes(); }); }; settingsOverlay.querySelector('#glm-reset-star').onclick = () => { const s = getUserSettings(); s.starIcon = null; saveUserSettings(s); previewStar.style.backgroundImage = `url('${DEFAULT_GEM_STAR}')`; lastRenderedSignature = ""; generateNavNodes(); }; function toggleListMode() { isListMode = !isListMode; if (isListMode) { wrapper.classList.add('list-mode'); btnView.innerHTML = '×'; btnView.title = "关闭列表"; } 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) { isCollapsed = forceState !== null ? forceState : !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)); btnTop.onclick = () => { if (currentQuestions.length > 0) scrollToQ(currentQuestions[0], true); }; btnBottom.onclick = () => { if (currentQuestions.length > 0) scrollToQ(currentQuestions[currentQuestions.length - 1], true); }; // [V14.15] 升级版点击逻辑 function scrollToQ(target, isManual) { if (isManual) { isClickScrolling = true; if (scrollDebounceTimer) clearTimeout(scrollDebounceTimer); // 看门狗:防止滚死了不解锁,2秒后强制解锁 setTimeout(() => { isClickScrolling = false; }, 2000); } // [V14.20 修复] 豆包强制使用 auto 跳转 const behavior = currentAdapter.name === 'Doubao' ? 'auto' : 'smooth'; target.scrollIntoView({ behavior: behavior, block: 'center' }); } // --- 渲染主函数 --- function generateNavNodes() { const validQuestions = currentAdapter.getQuestions(); 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 firstQText = currentAdapter.getText(validQuestions[0]); const lastQText = currentAdapter.getText(validQuestions[validQuestions.length-1]); const currentSignature = validQuestions.map(q => q.id).join('|') + `_${firstQText.slice(0,5)}_${lastQText.slice(0,5)}`; if (currentSignature === lastRenderedSignature) return; lastRenderedSignature = currentSignature; scrollArea.innerHTML = ''; const currentChatId = currentAdapter.getChatId(firstQText); const starredList = getStarredList(currentChatId); const currentIcons = getCurrentIcons(); validQuestions.forEach((q, index) => { const dot = document.createElement('div'); dot.className = 'glm-nav-dot'; // 确保 ID 存在 if (!q.id) q.id = `gem_auto_id_${index}`; dot.dataset.targetId = q.id; const isStarred = starredList.includes(q.id); if (currentIcons.star && currentIcons.star !== "null") dot.style.backgroundImage = isStarred ? `url(${currentIcons.star})` : `url(${currentIcons.normal})`; if(isStarred) dot.classList.add('is-starred'); let textRaw = currentAdapter.getText(q); 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 || isListMode) return; 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(); // 1. 立即锁定雷达 isClickScrolling = true; if (scrollDebounceTimer) clearTimeout(scrollDebounceTimer); // 2. 手动更新宝石状态 const allDots = scrollArea.querySelectorAll('.glm-nav-dot'); allDots.forEach(d => d.classList.remove('active')); dot.classList.add('active'); // 3. 滚动 const targetQ = document.getElementById(q.id); if (targetQ) { // [V14.20 修复] 豆包强制瞬间跳转 (auto) + 顶部对齐 (start),解决 React 容器滚动失效 if (currentAdapter.name === 'Doubao') { targetQ.scrollIntoView({ behavior: 'auto', block: 'start' }); } else { targetQ.scrollIntoView({ behavior: 'smooth', block: 'center' }); } // --- 智能判断高亮目标 --- let flashNode = targetQ; // ChatGPT/Grok/Doubao 高亮气泡 if (wrapper.classList.contains('site-chatgpt') || wrapper.classList.contains('site-grok') || wrapper.classList.contains('site-doubao')) { const bubble = targetQ.querySelector('.user-message-bubble-color') || targetQ.querySelector('[class*="bg-"]') || (wrapper.classList.contains('site-grok') ? targetQ : null) || (wrapper.classList.contains('site-doubao') ? targetQ : null); // 豆包容器本身即气泡 if (bubble) { flashNode = bubble; const r = window.getComputedStyle(bubble).borderRadius; if(r && r!=='0px') flashNode.style.setProperty('--glm-flash-radius', r); } } // 4. 触发高亮 document.querySelectorAll('.glm-flash-target').forEach(el => el.classList.remove('glm-flash-target')); flashNode.classList.remove('glm-flash-target'); void flashNode.offsetWidth; // 强制重绘 flashNode.classList.add('glm-flash-target'); } setTimeout(() => { isClickScrolling = false; }, 2000); }; dot.ondblclick = (e) => { e.stopPropagation(); const nowStarred = toggleStar(q.id, currentChatId); const freshIcons = getCurrentIcons(); if (freshIcons.star && freshIcons.star !== "null") dot.style.backgroundImage = nowStarred ? `url(${freshIcons.star})` : `url(${freshIcons.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); }); // ============================================================ // [核心] 主动雷达扫描引擎 (V14.15 智能休眠版) // ============================================================ if (document._gem_radar) { document.removeEventListener('scroll', document._gem_radar, true); } let radarTicking = false; document._gem_radar = () => { // [V14.15] 滚动防抖检测:只要在滚,就重置定时器 if (scrollDebounceTimer) clearTimeout(scrollDebounceTimer); scrollDebounceTimer = setTimeout(() => { // 100ms 没有滚动事件,视为滚动结束,解锁雷达 isClickScrolling = false; }, 100); // 如果正在点击跳转(被锁),或者正在计算中,直接退出 if (isClickScrolling) return; if (radarTicking) return; radarTicking = true; requestAnimationFrame(() => { if (!currentQuestions || currentQuestions.length === 0) { radarTicking = false; return; } const readingLine = window.innerHeight / 4; let closestQId = null; let minDistance = Infinity; for (const qData of currentQuestions) { const qNode = document.getElementById(qData.id); if (!qNode) continue; const rect = qNode.getBoundingClientRect(); const distance = Math.abs(rect.top - readingLine); if (distance < minDistance) { minDistance = distance; closestQId = qData.id; } } if (closestQId) { const activeDot = scrollArea.querySelector('.glm-nav-dot.active'); if (!activeDot || activeDot.dataset.targetId !== closestQId) { if (activeDot) activeDot.classList.remove('active'); const newActive = scrollArea.querySelector(`.glm-nav-dot[data-target-id="${closestQId}"]`); if (newActive) { newActive.classList.add('active'); const containerH = scrollArea.clientHeight; // 侧边栏跟随滚动 scrollArea.scrollTo({ top: newActive.offsetTop - (containerH / 2) + 12, behavior: 'smooth' }); } } } radarTicking = false; // [V14.20] 绘制连线 (仅当有2个以上时) const navList = document.getElementById('gem-nav-list'); const canvas = document.getElementById('gem-canvas'); if (!navList || !canvas) return; const ctx = canvas.getContext('2d'); canvas.width = 100; canvas.height = navList.scrollHeight || window.innerHeight; ctx.clearRect(0, 0, canvas.width, canvas.height); if (currentQuestions.length >= 2) { ctx.beginPath(); // [V14.21 修复] 豆包连线颜色逻辑:若是豆包(white bg),用深色线;其他用浅色线 const isDoubao = document.body.classList.contains('site-doubao'); ctx.strokeStyle = isDoubao ? 'rgba(0, 0, 0, 0.15)' : 'rgba(255, 255, 255, 0.15)'; ctx.lineWidth = 2; let first = true; Array.from(navList.children).forEach(item => { const y = item.offsetTop + (item.offsetHeight / 2); const x = 50; // 居中 if (first) { ctx.moveTo(x, y); first = false; } else { ctx.lineTo(x, y); } }); ctx.stroke(); } }); }; // [V14.20] 开启 Capture 模式,适配豆包内部滚动 document.addEventListener('scroll', document._gem_radar, { capture: true, passive: true }); } let timeout = null; const observer = new MutationObserver(() => { if (timeout) clearTimeout(timeout); timeout = setTimeout(generateNavNodes, 800); }); observer.observe(document.body, { childList: true, subtree: true }); setTimeout(generateNavNodes, 1000); })();