// ==UserScript== // @name ai对话历史快捷查询追溯 // @namespace https://bbs.tampermonkey.net.cn/ // @version 0.1.0 // @description 解决单个对话太长,难以追溯之前的问答的问题,提供对历史问题的快捷寻找 // @author 口吃者 // @match https://yuanbao.tencent.com/* // @match https://tongyi.aliyun.com/* // @match https://kimi.moonshot.cn/* // @match https://chat.deepseek.com/* // @require https://scriptcat.org/lib/2691/1.0.0/sweetalert2.all.min-11.15.10.js // @run-at document-end // @license MIT // ==/UserScript== (function () { 'use strict'; const questionSelectorOption = { 'yuanbao': '[data-conv-speaker="human"]', 'tongyi': '.questionItem--dS3Alcnv', 'kimi': '.user-content', 'deepseek': '.fbb737a4' }; const url = window.location.href; var questionSelector; if(url.includes('yuanbao')){ questionSelector = questionSelectorOption['yuanbao']; }else if(url.includes('tongyi')){ questionSelector = questionSelectorOption['tongyi']; }else if(url.includes('kimi')){ questionSelector = questionSelectorOption['kimi']; }else if(url.includes('deepseek')){ questionSelector = questionSelectorOption['deepseek']; } window.addEventListener('load', addPanel); // 辅助函数:平滑滚动到元素中心点 function scrollToElement(element) { return new Promise((resolve) => { element.scrollIntoView({ behavior: 'smooth', block: 'center' }); // 等待滚动动画完成 setTimeout(resolve, 500); // 假设滚动动画持续时间不超过500ms }); } /** * 根据选择器提取文本内容并裁剪到最大字符限制,并添加序号前缀 * @param {string} selector - CSS选择器字符串 * @param {number} maxChars - 每个提示项的最大字符数 * @returns {string[]} 裁剪后的提示项数组(带序号前缀且去除多余空格) */ function getTrimmedSuggestions(selector, maxChars) { // 参数验证 if (typeof selector !== 'string' || selector.trim() === '') { throw new Error('Invalid selector: must be a non-empty string'); } if (typeof maxChars !== 'number' || maxChars < 1) { throw new Error('Max chars must be a positive integer ≥1'); } // 获取所有匹配元素并裁剪文本 const elements = document.querySelectorAll(selector); const suggestions = []; elements.forEach(element => { // 新增:处理换行符和多余空格 const rawText = element.textContent; const cleanedText = rawText .replace(/[\r\n]+/g, ' ') // 替换换行符为空格 .replace(/\s+/g, ' ') // 合并连续空格 .trim(); // 移除首尾空格 if (cleanedText.length === 0) return; // 跳过空文本 // 裁剪逻辑 let trimmedText = cleanedText; if (cleanedText.length > maxChars) { trimmedText = cleanedText.slice(0, maxChars); // 可选:添加省略号(根据需求选择是否开启) // trimmedText += '...'; } suggestions.push(trimmedText); }); // 添加序号前缀 const result = []; const len = suggestions.length; for (let i = 0; i < len; i++) { const index = -(len - 1 - i); const prefix = `${index}:`; // 中文冒号 result.push(`${prefix}${suggestions[i]}`); } return result; } function addPanel() { function genButton(text, foo, id, fooParams = {}) { let b = document.createElement('button'); b.textContent = text; b.style.verticalAlign = 'inherit'; // 使用箭头函数创建闭包来保存 fooParams 并传递给 foo b.addEventListener('click', () => { foo.call(b, ...Object.values(fooParams)); // 使用 call 方法确保 this 指向按钮对象 }); if (id) { b.id = id }; return b; } function changeRangeDynamics() { const value = parseInt(this.value, 10); // 只能通过 DOM 方法改变 document.querySelector('#swal-range > output').textContent = value; } async function openPanelFunc() { var swalRangeValue = 0; // const promptArray = ["JavaScript框架", "React组件", "Vue开发"]; const { value: formValues } = await Swal.fire({ position: "center-end", draggable: true, backdrop: false, title: "Jump Location", showCancelButton: true, cancelButtonText: 'Cancel', confirmButtonText: 'Jump', //class="swal2-range" swalalert框架可能会对其有特殊处理,导致其内标签的id声明失效 html: `